mirror of https://github.com/procxx/kepka.git
				
				
				
			
		
			
				
	
	
		
			213 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			5.3 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
 | |
| 
 | |
| namespace Notify {
 | |
| struct PeerUpdate;
 | |
| } // namespace Notify
 | |
| class AudioMsgId;
 | |
| 
 | |
| namespace Media {
 | |
| namespace Player {
 | |
| 
 | |
| void start();
 | |
| void finish();
 | |
| 
 | |
| class Instance;
 | |
| Instance *instance();
 | |
| 
 | |
| struct TrackState;
 | |
| 
 | |
| class Instance : private base::Subscriber {
 | |
| public:
 | |
| 	void play(AudioMsgId::Type type);
 | |
| 	void pause(AudioMsgId::Type type);
 | |
| 	void stop(AudioMsgId::Type type);
 | |
| 	void playPause(AudioMsgId::Type type);
 | |
| 	bool next(AudioMsgId::Type type);
 | |
| 	bool previous(AudioMsgId::Type type);
 | |
| 
 | |
| 	AudioMsgId::Type getActiveType() const;
 | |
| 
 | |
| 	void play() {
 | |
| 		play(getActiveType());
 | |
| 	}
 | |
| 	void pause() {
 | |
| 		pause(getActiveType());
 | |
| 	}
 | |
| 	void stop() {
 | |
| 		stop(getActiveType());
 | |
| 	}
 | |
| 	void playPause() {
 | |
| 		playPause(getActiveType());
 | |
| 	}
 | |
| 	bool next() {
 | |
| 		return next(getActiveType());
 | |
| 	}
 | |
| 	bool previous() {
 | |
| 		return previous(getActiveType());
 | |
| 	}
 | |
| 
 | |
| 	void playPauseCancelClicked(AudioMsgId::Type type);
 | |
| 
 | |
| 	void play(const AudioMsgId &audioId);
 | |
| 	AudioMsgId current(AudioMsgId::Type type) const {
 | |
| 		if (auto data = getData(type)) {
 | |
| 			return data->current;
 | |
| 		}
 | |
| 		return AudioMsgId();
 | |
| 	}
 | |
| 
 | |
| 	bool repeatEnabled(AudioMsgId::Type type) const {
 | |
| 		if (auto data = getData(type)) {
 | |
| 			return data->repeatEnabled;
 | |
| 		}
 | |
| 		return false;
 | |
| 	}
 | |
| 	void toggleRepeat(AudioMsgId::Type type) {
 | |
| 		if (auto data = getData(type)) {
 | |
| 			data->repeatEnabled = !data->repeatEnabled;
 | |
| 			_repeatChangedNotifier.notify(type);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	bool isSeeking(AudioMsgId::Type type) const {
 | |
| 		if (auto data = getData(type)) {
 | |
| 			return (data->seeking == data->current);
 | |
| 		}
 | |
| 		return false;
 | |
| 	}
 | |
| 	void startSeeking(AudioMsgId::Type type);
 | |
| 	void stopSeeking(AudioMsgId::Type type);
 | |
| 
 | |
| 	QList<FullMsgId> playlist(AudioMsgId::Type type) const {
 | |
| 		if (auto data = getData(type)) {
 | |
| 			return data->playlist;
 | |
| 		}
 | |
| 		return QList<FullMsgId>();
 | |
| 	}
 | |
| 
 | |
| 	struct Switch {
 | |
| 		AudioMsgId from;
 | |
| 		FullMsgId to;
 | |
| 	};
 | |
| 
 | |
| 	base::Observable<Switch> &switchToNextNotifier() {
 | |
| 		return _switchToNextNotifier;
 | |
| 	}
 | |
| 	base::Observable<bool> &usePanelPlayer() {
 | |
| 		return _usePanelPlayer;
 | |
| 	}
 | |
| 	base::Observable<bool> &titleButtonOver() {
 | |
| 		return _titleButtonOver;
 | |
| 	}
 | |
| 	base::Observable<bool> &playerWidgetOver() {
 | |
| 		return _playerWidgetOver;
 | |
| 	}
 | |
| 	base::Observable<TrackState> &updatedNotifier() {
 | |
| 		return _updatedNotifier;
 | |
| 	}
 | |
| 	base::Observable<AudioMsgId::Type> &tracksFinishedNotifier() {
 | |
| 		return _tracksFinishedNotifier;
 | |
| 	}
 | |
| 	base::Observable<AudioMsgId::Type> &playlistChangedNotifier() {
 | |
| 		return _playlistChangedNotifier;
 | |
| 	}
 | |
| 	base::Observable<AudioMsgId::Type> &trackChangedNotifier() {
 | |
| 		return _trackChangedNotifier;
 | |
| 	}
 | |
| 	base::Observable<AudioMsgId::Type> &repeatChangedNotifier() {
 | |
| 		return _repeatChangedNotifier;
 | |
| 	}
 | |
| 
 | |
| 	void documentLoadProgress(DocumentData *document);
 | |
| 
 | |
| 	void clear();
 | |
| 
 | |
| private:
 | |
| 	Instance();
 | |
| 	friend void start();
 | |
| 
 | |
| 	struct Data {
 | |
| 		Data(AudioMsgId::Type type, MediaOverviewType overview) : type(type), overview(overview) {
 | |
| 		}
 | |
| 
 | |
| 		AudioMsgId::Type type;
 | |
| 		MediaOverviewType overview;
 | |
| 		AudioMsgId current;
 | |
| 		AudioMsgId seeking;
 | |
| 		History *history = nullptr;
 | |
| 		History *migrated = nullptr;
 | |
| 		bool repeatEnabled = false;
 | |
| 		QList<FullMsgId> playlist;
 | |
| 		bool isPlaying = false;
 | |
| 	};
 | |
| 
 | |
| 	// Observed notifications.
 | |
| 	void notifyPeerUpdated(const Notify::PeerUpdate &update);
 | |
| 	void handleSongUpdate(const AudioMsgId &audioId);
 | |
| 
 | |
| 	void checkPeerUpdate(AudioMsgId::Type type, const Notify::PeerUpdate &update);
 | |
| 	void setCurrent(const AudioMsgId &audioId);
 | |
| 	void rebuildPlaylist(Data *data);
 | |
| 	bool moveInPlaylist(Data *data, int delta, bool autonext);
 | |
| 	void preloadNext(Data *data);
 | |
| 	void handleLogout();
 | |
| 
 | |
| 	template <typename CheckCallback>
 | |
| 	void emitUpdate(AudioMsgId::Type type, CheckCallback check);
 | |
| 
 | |
| 	Data *getData(AudioMsgId::Type type) {
 | |
| 		if (type == AudioMsgId::Type::Song) {
 | |
| 			return &_songData;
 | |
| 		} else if (type == AudioMsgId::Type::Voice) {
 | |
| 			return &_voiceData;
 | |
| 		}
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 
 | |
| 	const Data *getData(AudioMsgId::Type type) const {
 | |
| 		if (type == AudioMsgId::Type::Song) {
 | |
| 			return &_songData;
 | |
| 		} else if (type == AudioMsgId::Type::Voice) {
 | |
| 			return &_voiceData;
 | |
| 		}
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 
 | |
| 	Data _songData;
 | |
| 	Data _voiceData;
 | |
| 
 | |
| 	base::Observable<Switch> _switchToNextNotifier;
 | |
| 	base::Observable<bool> _usePanelPlayer;
 | |
| 	base::Observable<bool> _titleButtonOver;
 | |
| 	base::Observable<bool> _playerWidgetOver;
 | |
| 	base::Observable<TrackState> _updatedNotifier;
 | |
| 	base::Observable<AudioMsgId::Type> _tracksFinishedNotifier;
 | |
| 	base::Observable<AudioMsgId::Type> _playlistChangedNotifier;
 | |
| 	base::Observable<AudioMsgId::Type> _trackChangedNotifier;
 | |
| 	base::Observable<AudioMsgId::Type> _repeatChangedNotifier;
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace Clip
 | |
| } // namespace Media
 |