560 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			560 lines
		
	
	
	
		
			13 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.
 | 
						|
 | 
						|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | 
						|
Copyright (c) 2014 John Preston, https://desktop.telegram.org
 | 
						|
*/
 | 
						|
#pragma once
 | 
						|
 | 
						|
typedef uint64 PeerId;
 | 
						|
typedef uint64 PhotoId;
 | 
						|
typedef uint64 VideoId;
 | 
						|
typedef uint64 AudioId;
 | 
						|
typedef uint64 DocumentId;
 | 
						|
typedef uint64 WebPageId;
 | 
						|
typedef int32 MsgId;
 | 
						|
 | 
						|
struct NotifySettings {
 | 
						|
	NotifySettings() : mute(0), sound("default"), previews(true), events(1) {
 | 
						|
	}
 | 
						|
	int32 mute;
 | 
						|
	string sound;
 | 
						|
	bool previews;
 | 
						|
	int32 events;
 | 
						|
};
 | 
						|
typedef NotifySettings *NotifySettingsPtr;
 | 
						|
 | 
						|
static const NotifySettingsPtr UnknownNotifySettings = NotifySettingsPtr(0);
 | 
						|
static const NotifySettingsPtr EmptyNotifySettings = NotifySettingsPtr(1);
 | 
						|
extern NotifySettings globalNotifyAll, globalNotifyUsers, globalNotifyChats;
 | 
						|
extern NotifySettingsPtr globalNotifyAllPtr, globalNotifyUsersPtr, globalNotifyChatsPtr;
 | 
						|
 | 
						|
inline bool isNotifyMuted(NotifySettingsPtr settings) {
 | 
						|
	if (settings == UnknownNotifySettings || settings == EmptyNotifySettings) {
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	return (settings->mute > unixtime());
 | 
						|
}
 | 
						|
 | 
						|
style::color peerColor(int32 index);
 | 
						|
ImagePtr userDefPhoto(int32 index);
 | 
						|
ImagePtr chatDefPhoto(int32 index);
 | 
						|
 | 
						|
struct ChatData;
 | 
						|
struct UserData;
 | 
						|
struct PeerData {
 | 
						|
	PeerData(const PeerId &id);
 | 
						|
	virtual ~PeerData() {
 | 
						|
		if (notify != UnknownNotifySettings && notify != EmptyNotifySettings) {
 | 
						|
			delete notify;
 | 
						|
			notify = UnknownNotifySettings;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	UserData *asUser();
 | 
						|
	const UserData *asUser() const;
 | 
						|
 | 
						|
	ChatData *asChat();
 | 
						|
	const ChatData *asChat() const;
 | 
						|
 | 
						|
	void updateName(const QString &newName, const QString &newNameOrPhone, const QString &newUsername);
 | 
						|
 | 
						|
	void fillNames();
 | 
						|
 | 
						|
	virtual void nameUpdated() {
 | 
						|
	}
 | 
						|
 | 
						|
	PeerId id;
 | 
						|
 | 
						|
	QString name;
 | 
						|
	QString nameOrPhone;
 | 
						|
	typedef QSet<QString> Names;
 | 
						|
	Names names; // for filtering
 | 
						|
	typedef QSet<QChar> NameFirstChars;
 | 
						|
	NameFirstChars chars;
 | 
						|
 | 
						|
	bool loaded;
 | 
						|
	bool chat;
 | 
						|
	uint64 access;
 | 
						|
	MTPinputPeer input;
 | 
						|
	MTPinputUser inputUser;
 | 
						|
 | 
						|
	int32 colorIndex;
 | 
						|
	style::color color;
 | 
						|
	ImagePtr photo;
 | 
						|
 | 
						|
	int32 nameVersion;
 | 
						|
 | 
						|
	NotifySettingsPtr notify;
 | 
						|
};
 | 
						|
static const uint64 UserNoAccess = 0xFFFFFFFFFFFFFFFFULL;
 | 
						|
 | 
						|
class PeerLink : public ITextLink {
 | 
						|
public:
 | 
						|
	PeerLink(PeerData *peer) : _peer(peer) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
	PeerData *peer() const {
 | 
						|
		return _peer;
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	PeerData *_peer;
 | 
						|
};
 | 
						|
 | 
						|
struct PhotoData;
 | 
						|
struct UserData : public PeerData {
 | 
						|
	UserData(const PeerId &id) : PeerData(id), lnk(new PeerLink(this)), onlineTill(0), contact(-1), photosCount(-1) {
 | 
						|
	}
 | 
						|
	void setPhoto(const MTPUserProfilePhoto &photo);
 | 
						|
	void setName(const QString &first, const QString &last, const QString &phoneName, const QString &username);
 | 
						|
	void setPhone(const QString &newPhone);
 | 
						|
	void nameUpdated();
 | 
						|
 | 
						|
	QString firstName;
 | 
						|
	QString lastName;
 | 
						|
	QString username;
 | 
						|
	QString phone;
 | 
						|
	Text nameText;
 | 
						|
	PhotoId photoId;
 | 
						|
	TextLinkPtr lnk;
 | 
						|
	int32 onlineTill;
 | 
						|
	int32 contact; // -1 - not contact, cant add (self, empty, deleted, foreign), 0 - not contact, can add (request), 1 - contact
 | 
						|
 | 
						|
	typedef QList<PhotoData*> Photos;
 | 
						|
	Photos photos;
 | 
						|
	int32 photosCount; // -1 not loaded, 0 all loaded
 | 
						|
};
 | 
						|
 | 
						|
struct ChatData : public PeerData {
 | 
						|
	ChatData(const PeerId &id) : PeerData(id), count(0), date(0), version(0), left(false), forbidden(true), photoId(0) {
 | 
						|
	}
 | 
						|
	void setPhoto(const MTPChatPhoto &photo, const PhotoId &phId = 0);
 | 
						|
	int32 count;
 | 
						|
	int32 date;
 | 
						|
	int32 version;
 | 
						|
	int32 admin;
 | 
						|
	bool left;
 | 
						|
	bool forbidden;
 | 
						|
	typedef QMap<UserData*, int32> Participants;
 | 
						|
	Participants participants;
 | 
						|
	typedef QMap<UserData*, bool> CanKick;
 | 
						|
	CanKick cankick;
 | 
						|
	typedef QList<UserData*> LastAuthors;
 | 
						|
	LastAuthors lastAuthors;
 | 
						|
	ImagePtr photoFull;
 | 
						|
	PhotoId photoId;
 | 
						|
	// geo
 | 
						|
};
 | 
						|
 | 
						|
typedef QMap<char, QPixmap> PreparedPhotoThumbs;
 | 
						|
struct PhotoData {
 | 
						|
	PhotoData(const PhotoId &id, const uint64 &access = 0, int32 user = 0, int32 date = 0, const ImagePtr &thumb = ImagePtr(), const ImagePtr &medium = ImagePtr(), const ImagePtr &full = ImagePtr()) :
 | 
						|
		id(id), access(access), user(user), date(date), thumb(thumb), medium(medium), full(full), chat(0) {
 | 
						|
	}
 | 
						|
	void forget() {
 | 
						|
		thumb->forget();
 | 
						|
		replyPreview->forget();
 | 
						|
		medium->forget();
 | 
						|
		full->forget();
 | 
						|
	}
 | 
						|
	ImagePtr makeReplyPreview() {
 | 
						|
		if (replyPreview->isNull() && !thumb->isNull()) {
 | 
						|
			if (thumb->loaded()) {
 | 
						|
				int w = thumb->width(), h = thumb->height();
 | 
						|
				if (w <= 0) w = 1;
 | 
						|
				if (h <= 0) h = 1;
 | 
						|
				replyPreview = ImagePtr(w > h ? thumb->pix(w * st::msgReplyBarSize.height() / h, st::msgReplyBarSize.height()) : thumb->pix(st::msgReplyBarSize.height()), "PNG");
 | 
						|
			} else {
 | 
						|
				thumb->load();
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return replyPreview;
 | 
						|
	}
 | 
						|
	PhotoId id;
 | 
						|
	uint64 access;
 | 
						|
	int32 user;
 | 
						|
	int32 date;
 | 
						|
	ImagePtr thumb, replyPreview;
 | 
						|
	ImagePtr medium;
 | 
						|
	ImagePtr full;
 | 
						|
	ChatData *chat; // for chat photos connection
 | 
						|
	// geo, caption
 | 
						|
 | 
						|
	int32 cachew;
 | 
						|
	QPixmap cache;
 | 
						|
};
 | 
						|
 | 
						|
class PhotoLink : public ITextLink {
 | 
						|
public:
 | 
						|
	PhotoLink(PhotoData *photo) : _photo(photo), _peer(0) {
 | 
						|
	}
 | 
						|
	PhotoLink(PhotoData *photo, PeerData *peer) : _photo(photo), _peer(peer) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
	PhotoData *photo() const {
 | 
						|
		return _photo;
 | 
						|
	}
 | 
						|
	PeerData *peer() const {
 | 
						|
		return _peer;
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	PhotoData *_photo;
 | 
						|
	PeerData *_peer;
 | 
						|
};
 | 
						|
 | 
						|
enum FileStatus {
 | 
						|
	FileFailed = -1,
 | 
						|
	FileUploading = 0,
 | 
						|
	FileReady = 1,
 | 
						|
};
 | 
						|
 | 
						|
struct VideoData {
 | 
						|
	VideoData(const VideoId &id, const uint64 &access = 0, int32 user = 0, int32 date = 0, int32 duration = 0, int32 w = 0, int32 h = 0, const ImagePtr &thumb = ImagePtr(), int32 dc = 0, int32 size = 0);
 | 
						|
 | 
						|
	void forget() {
 | 
						|
		thumb->forget();
 | 
						|
		replyPreview->forget();
 | 
						|
	}
 | 
						|
 | 
						|
	void save(const QString &toFile);
 | 
						|
 | 
						|
	void cancel(bool beforeDownload = false) {
 | 
						|
		mtpFileLoader *l = loader;
 | 
						|
		loader = 0;
 | 
						|
		if (l) {
 | 
						|
			l->cancel();
 | 
						|
			l->deleteLater();
 | 
						|
			l->rpcInvalidate();
 | 
						|
		}
 | 
						|
		location = FileLocation();
 | 
						|
		if (!beforeDownload) {
 | 
						|
			openOnSave = openOnSaveMsgId = 0;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	void finish() {
 | 
						|
		if (loader->done()) {
 | 
						|
			location = FileLocation(loader->fileType(), loader->fileName());
 | 
						|
		}
 | 
						|
		loader->deleteLater();
 | 
						|
		loader->rpcInvalidate();
 | 
						|
		loader = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	QString already(bool check = false);
 | 
						|
 | 
						|
	VideoId id;
 | 
						|
	uint64 access;
 | 
						|
	int32 user;
 | 
						|
	int32 date;
 | 
						|
	int32 duration;
 | 
						|
	int32 w, h;
 | 
						|
	ImagePtr thumb, replyPreview;
 | 
						|
	int32 dc, size;
 | 
						|
	// geo, caption
 | 
						|
 | 
						|
	FileStatus status;
 | 
						|
	int32 uploadOffset;
 | 
						|
 | 
						|
	mtpTypeId fileType;
 | 
						|
	int32 openOnSave, openOnSaveMsgId;
 | 
						|
	mtpFileLoader *loader;
 | 
						|
	FileLocation location;
 | 
						|
};
 | 
						|
 | 
						|
class VideoLink : public ITextLink {
 | 
						|
public:
 | 
						|
	VideoLink(VideoData *video) : _video(video) {
 | 
						|
	}
 | 
						|
	VideoData *video() const {
 | 
						|
		return _video;
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	VideoData *_video;
 | 
						|
};
 | 
						|
 | 
						|
class VideoSaveLink : public VideoLink {
 | 
						|
public:
 | 
						|
	VideoSaveLink(VideoData *video) : VideoLink(video) {
 | 
						|
	}
 | 
						|
	static void doSave(VideoData *video, bool forceSavingAs = false);
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
class VideoOpenLink : public VideoLink {
 | 
						|
public:
 | 
						|
	VideoOpenLink(VideoData *video) : VideoLink(video) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
class VideoCancelLink : public VideoLink {
 | 
						|
public:
 | 
						|
	VideoCancelLink(VideoData *video) : VideoLink(video) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
struct AudioData {
 | 
						|
	AudioData(const AudioId &id, const uint64 &access = 0, int32 user = 0, int32 date = 0, const QString &mime = QString(), int32 duration = 0, int32 dc = 0, int32 size = 0);
 | 
						|
 | 
						|
	void forget() {
 | 
						|
	}
 | 
						|
 | 
						|
	void save(const QString &toFile);
 | 
						|
 | 
						|
	void cancel(bool beforeDownload = false) {
 | 
						|
		mtpFileLoader *l = loader;
 | 
						|
		loader = 0;
 | 
						|
		if (l) {
 | 
						|
			l->cancel();
 | 
						|
			l->deleteLater();
 | 
						|
			l->rpcInvalidate();
 | 
						|
		}
 | 
						|
		location = FileLocation();
 | 
						|
		if (!beforeDownload) {
 | 
						|
			openOnSave = openOnSaveMsgId = 0;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	void finish() {
 | 
						|
		if (loader->done()) {
 | 
						|
			location = FileLocation(loader->fileType(), loader->fileName());
 | 
						|
			data = loader->bytes();
 | 
						|
		}
 | 
						|
		loader->deleteLater();
 | 
						|
		loader->rpcInvalidate();
 | 
						|
		loader = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	QString already(bool check = false);
 | 
						|
 | 
						|
	AudioId id;
 | 
						|
	uint64 access;
 | 
						|
	int32 user;
 | 
						|
	int32 date;
 | 
						|
	QString mime;
 | 
						|
	int32 duration;
 | 
						|
	int32 dc;
 | 
						|
	int32 size;
 | 
						|
 | 
						|
	FileStatus status;
 | 
						|
	int32 uploadOffset;
 | 
						|
 | 
						|
	int32 openOnSave, openOnSaveMsgId;
 | 
						|
	mtpFileLoader *loader;
 | 
						|
	FileLocation location;
 | 
						|
	QByteArray data;
 | 
						|
	int32 md5[8];
 | 
						|
};
 | 
						|
 | 
						|
class AudioLink : public ITextLink {
 | 
						|
public:
 | 
						|
	AudioLink(AudioData *audio) : _audio(audio) {
 | 
						|
	}
 | 
						|
	AudioData *audio() const {
 | 
						|
		return _audio;
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	AudioData *_audio;
 | 
						|
};
 | 
						|
 | 
						|
class AudioSaveLink : public AudioLink {
 | 
						|
public:
 | 
						|
	AudioSaveLink(AudioData *audio) : AudioLink(audio) {
 | 
						|
	}
 | 
						|
	static void doSave(AudioData *audio, bool forceSavingAs = false);
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
class AudioOpenLink : public AudioLink {
 | 
						|
public:
 | 
						|
	AudioOpenLink(AudioData *audio) : AudioLink(audio) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
class AudioCancelLink : public AudioLink {
 | 
						|
public:
 | 
						|
	AudioCancelLink(AudioData *audio) : AudioLink(audio) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
enum DocumentType {
 | 
						|
	FileDocument,
 | 
						|
	VideoDocument,
 | 
						|
	AudioDocument,
 | 
						|
	StickerDocument,
 | 
						|
	AnimatedDocument
 | 
						|
};
 | 
						|
struct DocumentData {
 | 
						|
	DocumentData(const DocumentId &id, const uint64 &access = 0, int32 date = 0, const QVector<MTPDocumentAttribute> &attributes = QVector<MTPDocumentAttribute>(), const QString &mime = QString(), const ImagePtr &thumb = ImagePtr(), int32 dc = 0, int32 size = 0);
 | 
						|
	void setattributes(const QVector<MTPDocumentAttribute> &attributes);
 | 
						|
 | 
						|
	void forget() {
 | 
						|
		thumb->forget();
 | 
						|
		sticker->forget();
 | 
						|
		replyPreview->forget();
 | 
						|
	}
 | 
						|
 | 
						|
	void save(const QString &toFile);
 | 
						|
 | 
						|
	void cancel(bool beforeDownload = false) {
 | 
						|
		mtpFileLoader *l = loader;
 | 
						|
		loader = 0;
 | 
						|
		if (l) {
 | 
						|
			l->cancel();
 | 
						|
			l->deleteLater();
 | 
						|
			l->rpcInvalidate();
 | 
						|
		}
 | 
						|
		location = FileLocation();
 | 
						|
		if (!beforeDownload) {
 | 
						|
			openOnSave = openOnSaveMsgId = 0;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	void finish() {
 | 
						|
		if (loader->done()) {
 | 
						|
			location = FileLocation(loader->fileType(), loader->fileName());
 | 
						|
			data = loader->bytes();
 | 
						|
		}
 | 
						|
		loader->deleteLater();
 | 
						|
		loader->rpcInvalidate();
 | 
						|
		loader = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	QString already(bool check = false);
 | 
						|
 | 
						|
	DocumentId id;
 | 
						|
	DocumentType type;
 | 
						|
	QSize dimensions;
 | 
						|
	int32 duration;
 | 
						|
	uint64 access;
 | 
						|
	int32 date;
 | 
						|
	QString name, mime, alt; // alt - for stickers
 | 
						|
	ImagePtr thumb, replyPreview;
 | 
						|
	int32 dc;
 | 
						|
	int32 size;
 | 
						|
 | 
						|
	FileStatus status;
 | 
						|
	int32 uploadOffset;
 | 
						|
 | 
						|
	int32 openOnSave, openOnSaveMsgId;
 | 
						|
	mtpFileLoader *loader;
 | 
						|
	FileLocation location;
 | 
						|
 | 
						|
	QByteArray data;
 | 
						|
	ImagePtr sticker;
 | 
						|
 | 
						|
	int32 md5[8];
 | 
						|
};
 | 
						|
 | 
						|
class DocumentLink : public ITextLink {
 | 
						|
public:
 | 
						|
	DocumentLink(DocumentData *document) : _document(document) {
 | 
						|
	}
 | 
						|
	DocumentData *document() const {
 | 
						|
		return _document;
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	DocumentData *_document;
 | 
						|
};
 | 
						|
 | 
						|
class DocumentSaveLink : public DocumentLink {
 | 
						|
public:
 | 
						|
	DocumentSaveLink(DocumentData *document) : DocumentLink(document) {
 | 
						|
	}
 | 
						|
	static void doSave(DocumentData *document, bool forceSavingAs = false);
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
class DocumentOpenLink : public DocumentLink {
 | 
						|
public:
 | 
						|
	DocumentOpenLink(DocumentData *document) : DocumentLink(document) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
class DocumentCancelLink : public DocumentLink {
 | 
						|
public:
 | 
						|
	DocumentCancelLink(DocumentData *document) : DocumentLink(document) {
 | 
						|
	}
 | 
						|
	void onClick(Qt::MouseButton button) const;
 | 
						|
};
 | 
						|
 | 
						|
enum WebPageType {
 | 
						|
	WebPagePhoto,
 | 
						|
	WebPageVideo,
 | 
						|
	WebPageProfile,
 | 
						|
	WebPageArticle
 | 
						|
};
 | 
						|
inline WebPageType toWebPageType(const QString &type) {
 | 
						|
	if (type == QLatin1String("photo")) return WebPagePhoto;
 | 
						|
	if (type == QLatin1String("video")) return WebPageVideo;
 | 
						|
	if (type == QLatin1String("profile")) return WebPageProfile;
 | 
						|
	return WebPageArticle;
 | 
						|
}
 | 
						|
 | 
						|
struct WebPageData {
 | 
						|
	WebPageData(const WebPageId &id, WebPageType type = WebPageArticle, const QString &url = QString(), const QString &displayUrl = QString(), const QString &siteName = QString(), const QString &title = QString(), const QString &description = QString(), PhotoData *photo = 0, int32 duration = 0, const QString &author = QString(), int32 pendingTill = 0);
 | 
						|
	
 | 
						|
	void forget() {
 | 
						|
		if (photo) photo->forget();
 | 
						|
	}
 | 
						|
 | 
						|
	WebPageId id;
 | 
						|
	WebPageType type;
 | 
						|
	QString url, displayUrl, siteName, title, description;
 | 
						|
	int32 duration;
 | 
						|
	QString author;
 | 
						|
	PhotoData *photo;
 | 
						|
	int32 pendingTill;
 | 
						|
};
 | 
						|
 | 
						|
QString saveFileName(const QString &title, const QString &filter, const QString &prefix, QString name, bool savingAs, const QDir &dir = QDir());
 | 
						|
MsgId clientMsgId();
 | 
						|
 | 
						|
struct MessageCursor {
 | 
						|
	MessageCursor() : position(0), anchor(0), scroll(QFIXED_MAX) {
 | 
						|
	}
 | 
						|
	MessageCursor(int position, int anchor, int scroll) : position(position), anchor(anchor), scroll(scroll) {
 | 
						|
	}
 | 
						|
	MessageCursor(const QTextEdit &edit) {
 | 
						|
		fillFrom(edit);
 | 
						|
	}
 | 
						|
	void fillFrom(const QTextEdit &edit) {
 | 
						|
		QTextCursor c = edit.textCursor();
 | 
						|
		position = c.position();
 | 
						|
		anchor = c.anchor();
 | 
						|
		QScrollBar *s = edit.verticalScrollBar();
 | 
						|
		scroll = s ? s->value() : QFIXED_MAX;
 | 
						|
	}
 | 
						|
	void applyTo(QTextEdit &edit, bool *lock = 0) {
 | 
						|
		if (lock) *lock = true;
 | 
						|
		QTextCursor c = edit.textCursor();
 | 
						|
		c.setPosition(anchor, QTextCursor::MoveAnchor);
 | 
						|
		c.setPosition(position, QTextCursor::KeepAnchor);
 | 
						|
		edit.setTextCursor(c);
 | 
						|
		QScrollBar *s = edit.verticalScrollBar();
 | 
						|
		if (s) s->setValue(scroll);
 | 
						|
		if (lock) *lock = false;
 | 
						|
	}
 | 
						|
	int position, anchor, scroll;
 | 
						|
};
 |