| 
									
										
										
										
											2016-06-09 14:51:24 +03:00
										 |  |  | /*
 | 
					
						
							|  |  |  | This file is part of Telegram Desktop, | 
					
						
							| 
									
										
										
										
											2018-01-03 13:23:14 +03:00
										 |  |  | the official desktop application for the Telegram messaging service. | 
					
						
							| 
									
										
										
										
											2016-06-09 14:51:24 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-03 13:23:14 +03:00
										 |  |  | For license and copyright information please follow this link: | 
					
						
							|  |  |  | https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | 
					
						
							| 
									
										
										
										
											2016-06-09 14:51:24 +03:00
										 |  |  | */ | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Data { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This module suggests a way to hold global data structures, that are
 | 
					
						
							|  |  |  | // created on demand and deleted at the end of the app launch.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Usage:
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // class MyData : public Data::AbstractStruct { .. data .. };
 | 
					
						
							|  |  |  | // Data::GlobalStructurePointer<MyData> myData;
 | 
					
						
							|  |  |  | // .. somewhere when needed ..
 | 
					
						
							|  |  |  | // myData.createIfNull();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class AbstractStructure { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	virtual ~AbstractStructure() = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | inline AbstractStructure::~AbstractStructure() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace internal { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void registerAbstractStructure(AbstractStructure **p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Must be created in global scope!
 | 
					
						
							|  |  |  |   // Structure is derived from AbstractStructure.
 | 
					
						
							|  |  |  | template <typename Structure> | 
					
						
							|  |  |  | class GlobalStructurePointer { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	GlobalStructurePointer() = default; | 
					
						
							|  |  |  | 	GlobalStructurePointer(const GlobalStructurePointer<Structure> &other) = delete; | 
					
						
							|  |  |  | 	GlobalStructurePointer &operator=(const GlobalStructurePointer<Structure> &other) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void createIfNull() { | 
					
						
							|  |  |  | 		if (!_p) { | 
					
						
							|  |  |  | 			_p = new Structure(); | 
					
						
							|  |  |  | 			internal::registerAbstractStructure(&_p); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	Structure *operator->() { | 
					
						
							| 
									
										
										
										
											2017-08-17 12:06:26 +03:00
										 |  |  | 		Assert(_p != nullptr); | 
					
						
							| 
									
										
										
										
											2016-06-09 14:51:24 +03:00
										 |  |  | 		return static_cast<Structure*>(_p); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const Structure *operator->() const { | 
					
						
							| 
									
										
										
										
											2017-08-17 12:06:26 +03:00
										 |  |  | 		Assert(_p != nullptr); | 
					
						
							| 
									
										
										
										
											2016-06-09 14:51:24 +03:00
										 |  |  | 		return static_cast<const Structure*>(_p); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	explicit operator bool() const { | 
					
						
							|  |  |  | 		return _p != nullptr; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  | 	AbstractStructure *_p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This method should be called at the end of the app launch.
 | 
					
						
							|  |  |  | // It will destroy all data structures created by Data::GlobalStructurePointer.
 | 
					
						
							|  |  |  | void clearGlobalStructures(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Data
 |