Media viewer:

Avatars are now viewable, with pagination
Fix border-radius transition
This commit is contained in:
morethanwords 2020-10-15 01:46:29 +03:00
parent cab8bc52cb
commit d91b97bc02
4 changed files with 165 additions and 69 deletions

View File

@ -7,7 +7,6 @@ import appImManager from "../lib/appManagers/appImManager";
import appMessagesManager from "../lib/appManagers/appMessagesManager"; import appMessagesManager from "../lib/appManagers/appMessagesManager";
import appPeersManager from "../lib/appManagers/appPeersManager"; import appPeersManager from "../lib/appManagers/appPeersManager";
import appPhotosManager from "../lib/appManagers/appPhotosManager"; import appPhotosManager from "../lib/appManagers/appPhotosManager";
import appProfileManager from "../lib/appManagers/appProfileManager";
import { logger } from "../lib/logger"; import { logger } from "../lib/logger";
import VideoPlayer from "../lib/mediaPlayer"; import VideoPlayer from "../lib/mediaPlayer";
import { RichTextProcessor } from "../lib/richtextprocessor"; import { RichTextProcessor } from "../lib/richtextprocessor";
@ -49,7 +48,7 @@ class AppMediaViewerBase<ContentAdditionType extends string, ButtonsAdditionType
public log: ReturnType<typeof logger>; public log: ReturnType<typeof logger>;
protected peerID = 0; protected isFirstOpen = true;
protected loadMediaPromiseUp: Promise<void> = null; protected loadMediaPromiseUp: Promise<void> = null;
protected loadMediaPromiseDown: Promise<void> = null; protected loadMediaPromiseDown: Promise<void> = null;
protected loadedAllMediaUp = false; protected loadedAllMediaUp = false;
@ -221,7 +220,6 @@ class AppMediaViewerBase<ContentAdditionType extends string, ButtonsAdditionType
if(this.setMoverAnimationPromise) return; if(this.setMoverAnimationPromise) return;
this.peerID = 0;
this.lazyLoadQueue.clear(); this.lazyLoadQueue.clear();
const promise = this.setMoverToTarget(this.lastTarget, true).then(({onAnimationEnd}) => onAnimationEnd); const promise = this.setMoverToTarget(this.lastTarget, true).then(({onAnimationEnd}) => onAnimationEnd);
@ -786,15 +784,14 @@ class AppMediaViewerBase<ContentAdditionType extends string, ButtonsAdditionType
this.setAuthorInfo(fromID, media.date); this.setAuthorInfo(fromID, media.date);
const isVideo = (media as MyDocument).type == 'video' || (media as MyDocument).type == 'gif'; const isVideo = (media as MyDocument).type == 'video' || (media as MyDocument).type == 'gif';
const isFirstOpen = !this.peerID;
if(isFirstOpen) { if(this.isFirstOpen) {
this.peerID = $rootScope.selectedPeerID;
//this.targetContainer = targetContainer; //this.targetContainer = targetContainer;
this.prevTargets = prevTargets; this.prevTargets = prevTargets;
this.nextTargets = nextTargets; this.nextTargets = nextTargets;
this.reverse = reverse; this.reverse = reverse;
this.needLoadMore = needLoadMore; this.needLoadMore = needLoadMore;
this.isFirstOpen = false;
//this.loadMore = loadMore; //this.loadMore = loadMore;
if(appSidebarRight.historyTabIDs.slice(-1)[0] == AppSidebarRight.SLIDERITEMSIDS.forward) { if(appSidebarRight.historyTabIDs.slice(-1)[0] == AppSidebarRight.SLIDERITEMSIDS.forward) {
@ -1073,8 +1070,9 @@ type AppMediaViewerTargetType = {
}; };
export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delete' | 'forward', AppMediaViewerTargetType> { export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delete' | 'forward', AppMediaViewerTargetType> {
public currentMessageID = 0; public currentMessageID = 0;
public peerID: number;
constructor() { constructor(private inputFilter: 'inputMessagesFilterPhotoVideo' | 'inputMessagesFilterChatPhotos' = 'inputMessagesFilterPhotoVideo') {
super(['delete', 'forward']); super(['delete', 'forward']);
const stub = document.createElement('div'); const stub = document.createElement('div');
@ -1110,16 +1108,17 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
this.author.container.addEventListener('click', this.onAuthorClick); this.author.container.addEventListener('click', this.onAuthorClick);
} }
public close(e?: MouseEvent) { /* public close(e?: MouseEvent) {
const good = !this.setMoverAnimationPromise; const good = !this.setMoverAnimationPromise;
const promise = super.close(e); const promise = super.close(e);
if(good) { // clear if(good) { // clear
this.currentMessageID = 0; this.currentMessageID = 0;
this.peerID = 0;
} }
return promise; return promise;
} } */
onPrevClick = (target: AppMediaViewerTargetType) => { onPrevClick = (target: AppMediaViewerTargetType) => {
this.nextTargets.unshift({element: this.lastTarget, mid: this.currentMessageID}); this.nextTargets.unshift({element: this.lastTarget, mid: this.currentMessageID});
@ -1138,7 +1137,7 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
}; };
onAuthorClick = (e: MouseEvent) => { onAuthorClick = (e: MouseEvent) => {
if(this.currentMessageID) { if(this.currentMessageID && this.currentMessageID != Number.MAX_SAFE_INTEGER) {
const mid = this.currentMessageID; const mid = this.currentMessageID;
this.close(e).then(() => { this.close(e).then(() => {
const message = appMessagesManager.getMessage(mid); const message = appMessagesManager.getMessage(mid);
@ -1168,8 +1167,8 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
protected loadMoreMedia = (older = true) => { protected loadMoreMedia = (older = true) => {
//if(!older && this.reverse) return; //if(!older && this.reverse) return;
if(older && this.loadedAllMediaDown) return; if(older && this.loadedAllMediaDown) return Promise.resolve();
else if(!older && this.loadedAllMediaUp) return; else if(!older && this.loadedAllMediaUp) return Promise.resolve();
if(older && this.loadMediaPromiseDown) return this.loadMediaPromiseDown; if(older && this.loadMediaPromiseDown) return this.loadMediaPromiseDown;
else if(!older && this.loadMediaPromiseUp) return this.loadMediaPromiseUp; else if(!older && this.loadMediaPromiseUp) return this.loadMediaPromiseUp;
@ -1191,7 +1190,7 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
const peerID = this.peerID; const peerID = this.peerID;
const promise = appMessagesManager.getSearch(peerID, '', const promise = appMessagesManager.getSearch(peerID, '',
{_: 'inputMessagesFilterPhotoVideo'}, maxID, loadCount/* older ? loadCount : 0 */, 0, backLimit).then(value => { {_: this.inputFilter}, maxID, loadCount/* older ? loadCount : 0 */, 0, backLimit).then(value => {
if(this.peerID != peerID) { if(this.peerID != peerID) {
this.log.warn('peer changed'); this.log.warn('peer changed');
return; return;
@ -1212,10 +1211,10 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
const method = older ? value.history.forEach : value.history.forEachReverse; const method = older ? value.history.forEach : value.history.forEachReverse;
method.call(value.history, mid => { method.call(value.history, mid => {
const message = appMessagesManager.getMessage(mid); const message = appMessagesManager.getMessage(mid);
const media = message.media; const media = this.getMediaFromMessage(message);
if(!media || !(media.photo || media.document || (media.webpage && media.webpage.document))) return; if(!media) return;
if(media._ == 'document' && media.type != 'video') return; //if(media._ == 'document' && media.type != 'video') return;
const t = {element: null as HTMLElement, mid: mid}; const t = {element: null as HTMLElement, mid: mid};
if(older) { if(older) {
@ -1227,8 +1226,8 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
} }
}); });
this.buttons.prev.style.display = this.prevTargets.length ? '' : 'none'; this.buttons.prev.classList.toggle('hide', !this.prevTargets.length);
this.buttons.next.style.display = this.nextTargets.length ? '' : 'none'; this.buttons.next.classList.toggle('hide', !this.nextTargets.length);
}, () => {}).then(() => { }, () => {}).then(() => {
if(older) this.loadMediaPromiseDown = null; if(older) this.loadMediaPromiseDown = null;
else this.loadMediaPromiseUp = null; else this.loadMediaPromiseUp = null;
@ -1240,6 +1239,12 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
return promise; return promise;
}; };
private getMediaFromMessage(message: any) {
return message.action ? message.action.photo : message.media.photo
|| message.media.document
|| (message.media.webpage && (message.media.webpage.document || message.media.webpage.photo));
}
private setCaption(message: any) { private setCaption(message: any) {
const caption = message.message; const caption = message.message;
if(caption) { if(caption) {
@ -1255,20 +1260,21 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
prevTargets: AppMediaViewer['prevTargets'] = [], nextTargets: AppMediaViewer['prevTargets'] = [], needLoadMore = true) { prevTargets: AppMediaViewer['prevTargets'] = [], nextTargets: AppMediaViewer['prevTargets'] = [], needLoadMore = true) {
if(this.setMoverPromise) return this.setMoverPromise; if(this.setMoverPromise) return this.setMoverPromise;
const mid = message.mid;
const fromID = message.fromID; const fromID = message.fromID;
const media = message.media.photo || message.media.document || message.media.webpage.document || message.media.webpage.photo; const media = this.getMediaFromMessage(message);
const isFirstOpen = !this.peerID;
let fromRight = 0; let fromRight = 0;
if(!isFirstOpen) { if(!this.isFirstOpen) {
//if(this.lastTarget === prevTarget) { //if(this.lastTarget === prevTarget) {
if(this.reverse) fromRight = this.currentMessageID < message.mid ? 1 : -1; if(this.reverse) fromRight = this.currentMessageID < mid ? 1 : -1;
else fromRight = this.currentMessageID > message.mid ? 1 : -1; else fromRight = this.currentMessageID > mid ? 1 : -1;
} else { } else {
this.reverse = reverse; this.reverse = reverse;
this.peerID = $rootScope.selectedPeerID;
} }
this.currentMessageID = message.mid; this.currentMessageID = mid;
const promise = super._openMedia(media, fromID, fromRight, target, reverse, prevTargets, nextTargets, needLoadMore); const promise = super._openMedia(media, fromID, fromRight, target, reverse, prevTargets, nextTargets, needLoadMore);
this.setCaption(message); this.setCaption(message);
@ -1276,10 +1282,16 @@ export default class AppMediaViewer extends AppMediaViewerBase<'caption', 'delet
} }
} }
export class AppMediaViewerAvatar extends AppMediaViewerBase<'', 'delete', {}> { type AppMediaViewerAvatarTargetType = {element: HTMLElement, photoID: string};
constructor() { export class AppMediaViewerAvatar extends AppMediaViewerBase<'', 'delete', AppMediaViewerAvatarTargetType> {
public currentPhotoID: string;
public peerID: number;
constructor(peerID: number) {
super(['delete']); super(['delete']);
this.peerID = peerID;
this.setBtnMenuToggle([{ this.setBtnMenuToggle([{
icon: 'download', icon: 'download',
text: 'Download', text: 'Download',
@ -1295,33 +1307,63 @@ export class AppMediaViewerAvatar extends AppMediaViewerBase<'', 'delete', {}> {
this.setListeners(); this.setListeners();
} }
onPrevClick = (target: AppMediaViewerAvatarTargetType) => {
this.nextTargets.unshift({element: this.lastTarget, photoID: this.currentPhotoID});
this.openMedia(target.photoID, target.element, -1);
};
onNextClick = (target: AppMediaViewerAvatarTargetType) => {
this.prevTargets.push({element: this.lastTarget, photoID: this.currentPhotoID});
this.openMedia(target.photoID, target.element, 1);
};
onDownloadClick = (e: MouseEvent) => { onDownloadClick = (e: MouseEvent) => {
}; };
protected loadMoreMedia = (older = true) => { protected loadMoreMedia = (older = true) => {
return Promise.resolve(); if(this.peerID < 0) return Promise.resolve(); // ! это значит, что открыло аватар чата, но следующих фотографий нет.
}; if(this.loadedAllMediaDown) return Promise.resolve();
if(this.loadMediaPromiseDown) return this.loadMediaPromiseDown;
public async openMedia(peerID: number, target?: HTMLElement, reverse = false, const peerID = this.peerID;
prevTargets: AppMediaViewerAvatar['prevTargets'] = [], nextTargets: AppMediaViewerAvatar['prevTargets'] = [], needLoadMore = true) { const loadCount = 50;
if(this.setMoverPromise) return this.setMoverPromise;
return appProfileManager.getFullPhoto(peerID).then(photo => { const maxID = this.nextTargets.length ? this.nextTargets[this.nextTargets.length - 1].photoID : this.currentPhotoID;
const fromID = peerID;
const isFirstOpen = !this.peerID; const promise = appPhotosManager.getUserPhotos(peerID, maxID, loadCount).then(value => {
let fromRight = 0; if(this.peerID != peerID) {
if(!isFirstOpen) { this.log.warn('peer changed');
//if(this.lastTarget === prevTarget) { return;
/* if(this.reverse) fromRight = this.currentMessageID < message.mid ? 1 : -1;
else fromRight = this.currentMessageID > message.mid ? 1 : -1; */
fromRight = 1;
} else {
this.reverse = reverse;
} }
const promise = super._openMedia(photo, fromID, fromRight, target, reverse, prevTargets, nextTargets, needLoadMore); this.log('loaded more media by maxID:', /* maxID, */value, older, this.reverse);
if(value.photos.length < loadCount) {
this.loadedAllMediaDown = true;
}
value.photos.forEach(photoID => {
if(this.currentPhotoID == photoID) return;
this.nextTargets.push({element: null as HTMLElement, photoID});
});
this.buttons.prev.classList.toggle('hide', !this.prevTargets.length);
this.buttons.next.classList.toggle('hide', !this.nextTargets.length);
}, () => {}).then(() => {
this.loadMediaPromiseDown = null;
}); });
return this.loadMediaPromiseDown = promise;
};
public async openMedia(photoID: string, target?: HTMLElement, fromRight = 0) {
if(this.setMoverPromise) return this.setMoverPromise;
const photo = appPhotosManager.getPhoto(photoID);
this.currentPhotoID = photo.id;
return super._openMedia(photo, this.peerID, fromRight, target, false);
} }
} }

View File

@ -1,7 +1,8 @@
import appMessagesManager from "../lib/appManagers/appMessagesManager";
import appProfileManager from "../lib/appManagers/appProfileManager"; import appProfileManager from "../lib/appManagers/appProfileManager";
import $rootScope from "../lib/rootScope"; import $rootScope from "../lib/rootScope";
import { cancelEvent } from "../lib/utils"; import { cancelEvent } from "../lib/utils";
import { AppMediaViewerAvatar } from "./appMediaViewer"; import AppMediaViewer, { AppMediaViewerAvatar } from "./appMediaViewer";
$rootScope.$on('avatar_update', (e) => { $rootScope.$on('avatar_update', (e) => {
let peerID = e.detail; let peerID = e.detail;
@ -30,17 +31,60 @@ export default class AvatarElement extends HTMLElement {
this.isDialog = !!this.getAttribute('dialog'); this.isDialog = !!this.getAttribute('dialog');
if(this.getAttribute('clickable') === '') { if(this.getAttribute('clickable') === '') {
this.setAttribute('clickable', 'set'); this.setAttribute('clickable', 'set');
this.addEventListener('click', (e) => { let loading = false;
this.addEventListener('click', async(e) => {
cancelEvent(e); cancelEvent(e);
if(loading) return;
//console.log('avatar clicked'); //console.log('avatar clicked');
const peerID = this.peerID; const peerID = this.peerID;
appProfileManager.getFullPhoto(this.peerID).then(photo => { loading = true;
if(this.peerID != peerID) return;
if(photo) { const photo = await appProfileManager.getFullPhoto(this.peerID);
if(this.peerID != peerID || !photo) {
loading = false;
return;
}
if(peerID < 0) {
const maxID = Number.MAX_SAFE_INTEGER;
const inputFilter = 'inputMessagesFilterChatPhotos';
const mid = await appMessagesManager.getSearch(peerID, '', {_: inputFilter}, maxID, 2, 0, 1).then(value => {
//console.log(lol);
// ! by descend
return value.history[0];
});
if(mid) {
// ! гений в деле, костылируем (но это гениально)
let message = appMessagesManager.getMessage(mid);
const messagePhoto = message.action.photo;
if(messagePhoto.id != photo.id) {
message = {
_: 'message',
mid: maxID,
media: {
_: 'messageMediaPhoto',
photo: photo
},
fromID: peerID
};
appMessagesManager.messagesStorage[maxID] = message;
}
const good = Array.from(this.querySelectorAll('img')).find(img => !img.classList.contains('emoji')); const good = Array.from(this.querySelectorAll('img')).find(img => !img.classList.contains('emoji'));
new AppMediaViewerAvatar().openMedia(peerID, good ? this : null); new AppMediaViewer(inputFilter).openMedia(message, good ? this : null);
loading = false;
return;
} }
}); }
if(photo) {
const good = Array.from(this.querySelectorAll('img')).find(img => !img.classList.contains('emoji'));
new AppMediaViewerAvatar(peerID).openMedia(photo.id, good ? this : null);
}
loading = false;
}); });
} }
} }

View File

@ -448,7 +448,8 @@ type MyInputMessagesFilter = 'inputMessagesFilterEmpty'
| 'inputMessagesFilterRoundVideo' | 'inputMessagesFilterRoundVideo'
| 'inputMessagesFilterMusic' | 'inputMessagesFilterMusic'
| 'inputMessagesFilterUrl' | 'inputMessagesFilterUrl'
| 'inputMessagesFilterMyMentions'; | 'inputMessagesFilterMyMentions'
| 'inputMessagesFilterChatPhotos';
export class AppMessagesManager { export class AppMessagesManager {
public messagesStorage: {[mid: string]: any} = {}; public messagesStorage: {[mid: string]: any} = {};
@ -2314,13 +2315,14 @@ export class AppMessagesManager {
} }
if(apiMessage.action) { if(apiMessage.action) {
let migrateFrom; let migrateFrom: number;
let migrateTo; let migrateTo: number;
switch(apiMessage.action._) { switch(apiMessage.action._) {
//case 'messageActionChannelEditPhoto':
case 'messageActionChatEditPhoto': case 'messageActionChatEditPhoto':
apiMessage.action.photo = appPhotosManager.savePhoto(apiMessage.action.photo, mediaContext); apiMessage.action.photo = appPhotosManager.savePhoto(apiMessage.action.photo, mediaContext);
//appPhotosManager.savePhoto(apiMessage.action.photo, mediaContext); //appPhotosManager.savePhoto(apiMessage.action.photo, mediaContext);
if(isBroadcast) { if(isBroadcast) { // ! messageActionChannelEditPhoto не существует в принципе, это используется для перевода.
apiMessage.action._ = 'messageActionChannelEditPhoto'; apiMessage.action._ = 'messageActionChannelEditPhoto';
} }
break; break;
@ -2947,6 +2949,10 @@ export class AppMessagesManager {
neededContents['url'] = true; neededContents['url'] = true;
break; break;
case 'inputMessagesFilterChatPhotos':
neededContents['avatar'] = true;
break;
/* case 'inputMessagesFilterMyMentions': /* case 'inputMessagesFilterMyMentions':
neededContents['mentioned'] = true; neededContents['mentioned'] = true;
break; */ break; */
@ -2975,6 +2981,8 @@ export class AppMessagesManager {
found = true; found = true;
} else if(neededContents['url'] && message.message && RichTextProcessor.matchUrl(message.message)) { } else if(neededContents['url'] && message.message && RichTextProcessor.matchUrl(message.message)) {
found = true; found = true;
} else if(neededContents['avatar'] && message.action && ['messageActionChannelEditPhoto', 'messageActionChatEditPhoto'].includes(message.action._)) {
found = true;
} }
if(found) { if(found) {
@ -4402,6 +4410,8 @@ export class AppMessagesManager {
} else if(this.needSingleMessages.indexOf(msgID) == -1) { } else if(this.needSingleMessages.indexOf(msgID) == -1) {
this.needSingleMessages.push(msgID); this.needSingleMessages.push(msgID);
return this.fetchSingleMessages(); return this.fetchSingleMessages();
} else if(this.fetchSingleMessagesPromise) {
return this.fetchSingleMessagesPromise;
} }
} }

View File

@ -1,12 +1,14 @@
import { CancellablePromise } from "../../helpers/cancellablePromise"; import { CancellablePromise } from "../../helpers/cancellablePromise";
import { isSafari } from "../../helpers/userAgent"; import { isSafari } from "../../helpers/userAgent";
import { FileLocation, InputFileLocation, Photo, PhotoSize } from "../../layer"; import { FileLocation, InputFileLocation, Photo, PhotoSize, PhotosPhotos } from "../../layer";
import { bytesFromHex, getFileNameByLocation } from "../bin_utils"; import { bytesFromHex, getFileNameByLocation } from "../bin_utils";
import { DownloadOptions } from "../mtproto/apiFileManager"; import { DownloadOptions } from "../mtproto/apiFileManager";
import apiManager from "../mtproto/mtprotoworker";
import referenceDatabase, { ReferenceContext } from "../mtproto/referenceDatabase"; import referenceDatabase, { ReferenceContext } from "../mtproto/referenceDatabase";
import { calcImageInBox, isObject, safeReplaceArrayInObject } from "../utils"; import { calcImageInBox, isObject, safeReplaceArrayInObject } from "../utils";
import { MyDocument } from "./appDocsManager"; import { MyDocument } from "./appDocsManager";
import appDownloadManager from "./appDownloadManager"; import appDownloadManager from "./appDownloadManager";
import appUsersManager from "./appUsersManager";
export type MyPhoto = Photo.photo; export type MyPhoto = Photo.photo;
@ -108,29 +110,27 @@ export class AppPhotosManager {
return bestPhotoSize; return bestPhotoSize;
} }
/* public getUserPhotos(userID: number, maxID: number, limit: number) { public getUserPhotos(userID: number, maxID: string = '0', limit: number = 20) {
var inputUser = appUsersManager.getUserInput(userID); const inputUser = appUsersManager.getUserInput(userID);
return apiManager.invokeApi('photos.getUserPhotos', { return apiManager.invokeApi('photos.getUserPhotos', {
user_id: inputUser, user_id: inputUser,
offset: 0, offset: 0,
limit: limit || 20, limit: limit,
max_id: maxID || 0 max_id: maxID
}).then((photosResult: any) => { }).then((photosResult) => {
appUsersManager.saveApiUsers(photosResult.users); appUsersManager.saveApiUsers(photosResult.users);
var photoIDs = []; const photoIDs: string[] = [];
var context = {user_id: userID}; photosResult.photos.forEach((photo, idx) => {
for(var i = 0; i < photosResult.photos.length; i++) { photosResult.photos[idx] = this.savePhoto(photo, {type: 'profilePhoto', peerID: userID});
//this.savePhoto(photosResult.photos[i], context); photoIDs.push(photo.id);
photosResult.photos[i] = this.savePhoto(photosResult.photos[i], context); });
photoIDs.push(photosResult.photos[i].id);
}
return { return {
count: photosResult.count || photosResult.photos.length, count: (photosResult as PhotosPhotos.photosPhotosSlice).count || photosResult.photos.length,
photos: photoIDs photos: photoIDs
}; };
}); });
} */ }
public getPreviewURLFromBytes(bytes: Uint8Array | number[], isSticker = false) { public getPreviewURLFromBytes(bytes: Uint8Array | number[], isSticker = false) {
let arr: Uint8Array; let arr: Uint8Array;