159 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| 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 <QtCore/QMap>
 | |
| 
 | |
| // ordered set template based on QMap
 | |
| template <typename T>
 | |
| class OrderedSet {
 | |
| 	struct NullType {
 | |
| 	};
 | |
| 	using Self = OrderedSet<T>;
 | |
| 	using Impl = QMap<T, NullType>;
 | |
| 	using IteratorImpl = typename Impl::iterator;
 | |
| 	using ConstIteratorImpl = typename Impl::const_iterator;
 | |
| 	Impl impl_;
 | |
| 
 | |
| public:
 | |
| 	inline bool operator==(const Self &other) const { return impl_ == other.impl_; }
 | |
| 	inline bool operator!=(const Self &other) const { return impl_ != other.impl_; }
 | |
| 	inline int size() const { return impl_.size(); }
 | |
| 	inline bool isEmpty() const { return impl_.isEmpty(); }
 | |
| 	inline void detach() { return impl_.detach(); }
 | |
| 	inline bool isDetached() const { return impl_.isDetached(); }
 | |
| 	inline void clear() { return impl_.clear(); }
 | |
| 	inline QList<T> values() const { return impl_.keys(); }
 | |
| 	inline const T &first() const { return impl_.firstKey(); }
 | |
| 	inline const T &last() const { return impl_.lastKey(); }
 | |
| 
 | |
| 	class const_iterator;
 | |
| 	class iterator {
 | |
| 	public:
 | |
| 		typedef typename IteratorImpl::iterator_category iterator_category;
 | |
| 		typedef typename IteratorImpl::difference_type difference_type;
 | |
| 		typedef T value_type;
 | |
| 		typedef T *pointer;
 | |
| 		typedef T &reference;
 | |
| 
 | |
| 		iterator() = default;
 | |
| 		iterator(const iterator &other) = default;
 | |
| 		iterator &operator=(const iterator &other) = default;
 | |
| 		inline const T &operator*() const { return impl_.key(); }
 | |
| 		inline const T *operator->() const { return &impl_.key(); }
 | |
| 		inline bool operator==(const iterator &other) const { return impl_ == other.impl_; }
 | |
| 		inline bool operator!=(const iterator &other) const { return impl_ != other.impl_; }
 | |
| 		inline iterator &operator++() { ++impl_; return *this; }
 | |
| 		inline iterator operator++(int) { return iterator(impl_++); }
 | |
| 		inline iterator &operator--() { --impl_; return *this; }
 | |
| 		inline iterator operator--(int) { return iterator(impl_--); }
 | |
| 		inline iterator operator+(int j) const { return iterator(impl_ + j); }
 | |
| 		inline iterator operator-(int j) const { return iterator(impl_ - j); }
 | |
| 		inline iterator &operator+=(int j) { impl_ += j; return *this; }
 | |
| 		inline iterator &operator-=(int j) { impl_ -= j; return *this; }
 | |
| 
 | |
| 		friend class const_iterator;
 | |
| 		inline bool operator==(const const_iterator &other) const { return impl_ == other.impl_; }
 | |
| 		inline bool operator!=(const const_iterator &other) const { return impl_ != other.impl_; }
 | |
| 
 | |
| 	private:
 | |
| 		explicit iterator(const IteratorImpl &impl) : impl_(impl) {
 | |
| 		}
 | |
| 		IteratorImpl impl_;
 | |
| 		friend class OrderedSet<T>;
 | |
| 
 | |
| 	};
 | |
| 	friend class iterator;
 | |
| 
 | |
| 	class const_iterator {
 | |
| 	public:
 | |
| 		typedef typename IteratorImpl::iterator_category iterator_category;
 | |
| 		typedef typename IteratorImpl::difference_type difference_type;
 | |
| 		typedef T value_type;
 | |
| 		typedef T *pointer;
 | |
| 		typedef T &reference;
 | |
| 
 | |
| 		const_iterator() = default;
 | |
| 		const_iterator(const const_iterator &other) = default;
 | |
| 		const_iterator &operator=(const const_iterator &other) = default;
 | |
| 		const_iterator(const iterator &other) : impl_(other.impl_) {
 | |
| 		}
 | |
| 		const_iterator &operator=(const iterator &other) {
 | |
| 			impl_ = other.impl_;
 | |
| 			return *this;
 | |
| 		}
 | |
| 		inline const T &operator*() const { return impl_.key(); }
 | |
| 		inline const T *operator->() const { return &impl_.key(); }
 | |
| 		inline bool operator==(const const_iterator &other) const { return impl_ == other.impl_; }
 | |
| 		inline bool operator!=(const const_iterator &other) const { return impl_ != other.impl_; }
 | |
| 		inline const_iterator &operator++() { ++impl_; return *this; }
 | |
| 		inline const_iterator operator++(int) { return const_iterator(impl_++); }
 | |
| 		inline const_iterator &operator--() { --impl_; return *this; }
 | |
| 		inline const_iterator operator--(int) { return const_iterator(impl_--); }
 | |
| 		inline const_iterator operator+(int j) const { return const_iterator(impl_ + j); }
 | |
| 		inline const_iterator operator-(int j) const { return const_iterator(impl_ - j); }
 | |
| 		inline const_iterator &operator+=(int j) { impl_ += j; return *this; }
 | |
| 		inline const_iterator &operator-=(int j) { impl_ -= j; return *this; }
 | |
| 
 | |
| 		friend class iterator;
 | |
| 		inline bool operator==(const iterator &other) const { return impl_ == other.impl_; }
 | |
| 		inline bool operator!=(const iterator &other) const { return impl_ != other.impl_; }
 | |
| 
 | |
| 	private:
 | |
| 		explicit const_iterator(const ConstIteratorImpl &impl) : impl_(impl) {
 | |
| 		}
 | |
| 		ConstIteratorImpl impl_;
 | |
| 		friend class OrderedSet<T>;
 | |
| 
 | |
| 	};
 | |
| 	friend class const_iterator;
 | |
| 
 | |
| 	// STL style
 | |
| 	inline iterator begin() { return iterator(impl_.begin()); }
 | |
| 	inline const_iterator begin() const { return const_iterator(impl_.cbegin()); }
 | |
| 	inline const_iterator constBegin() const { return const_iterator(impl_.cbegin()); }
 | |
| 	inline const_iterator cbegin() const { return const_iterator(impl_.cbegin()); }
 | |
| 	inline iterator end() { detach(); return iterator(impl_.end()); }
 | |
| 	inline const_iterator end() const { return const_iterator(impl_.cend()); }
 | |
| 	inline const_iterator constEnd() const { return const_iterator(impl_.cend()); }
 | |
| 	inline const_iterator cend() const { return const_iterator(impl_.cend()); }
 | |
| 	inline iterator erase(iterator it) { return iterator(impl_.erase(it.impl_)); }
 | |
| 
 | |
| 	inline iterator insert(const T &value) { return iterator(impl_.insert(value, NullType())); }
 | |
| 	inline iterator insert(const_iterator pos, const T &value) { return iterator(impl_.insert(pos.impl_, value, NullType())); }
 | |
| 	inline int remove(const T &value) { return impl_.remove(value); }
 | |
| 	inline bool contains(const T &value) const { return impl_.contains(value); }
 | |
| 
 | |
| 	// more Qt
 | |
| 	typedef iterator Iterator;
 | |
| 	typedef const_iterator ConstIterator;
 | |
| 	inline int count() const { return impl_.count(); }
 | |
| 	inline iterator find(const T &value) { return iterator(impl_.find(value)); }
 | |
| 	inline const_iterator find(const T &value) const { return const_iterator(impl_.constFind(value)); }
 | |
| 	inline const_iterator constFind(const T &value) const { return const_iterator(impl_.constFind(value)); }
 | |
| 	inline Self &unite(const Self &other) { impl_.unite(other.impl_); return *this; }
 | |
| 
 | |
| 	// STL compatibility
 | |
| 	typedef typename Impl::difference_type difference_type;
 | |
| 	typedef typename Impl::size_type size_type;
 | |
| 	inline bool empty() const { return impl_.empty(); }
 | |
| 
 | |
| };
 | 
