Avatar stripped thumbnails

Render avatar thumb if have queue
This commit is contained in:
morethanwords 2021-06-01 18:20:03 +03:00
parent 6a24ee2407
commit bdc4851210
4 changed files with 67 additions and 39 deletions

View File

@ -179,6 +179,22 @@ export default class AvatarElement extends HTMLElement {
} }
} }
private r(onlyThumb = false) {
const res = appProfileManager.putPhoto(this, this.peerId, this.isDialog, this.peerTitle, onlyThumb);
const promise = res ? res.loadPromise : Promise.resolve();
if(this.loadPromises) {
if(res && res.cached) {
this.loadPromises.push(promise);
}
promise.finally(() => {
this.loadPromises = undefined;
});
}
return res;
}
public update() { public update() {
if(this.lazyLoadQueue) { if(this.lazyLoadQueue) {
if(!seen.has(this.peerId)) { if(!seen.has(this.peerId)) {
@ -193,6 +209,8 @@ export default class AvatarElement extends HTMLElement {
set.add(this); set.add(this);
this.r(true);
this.lazyLoadQueue.push({ this.lazyLoadQueue.push({
div: this, div: this,
load: () => { load: () => {
@ -209,17 +227,8 @@ export default class AvatarElement extends HTMLElement {
seen.add(this.peerId); seen.add(this.peerId);
const res = appProfileManager.putPhoto(this, this.peerId, this.isDialog, this.peerTitle); const res = this.r();
const promise = res ? res.loadPromise : Promise.resolve(); const promise = res ? res.loadPromise : Promise.resolve();
if(this.loadPromises) {
if(res && res.cached) {
this.loadPromises.push(promise);
}
promise.finally(() => {
this.loadPromises = undefined;
});
}
if(this.addedToQueue) { if(this.addedToQueue) {
promise.finally(() => { promise.finally(() => {

View File

@ -12,11 +12,11 @@ const set = (elem: HTMLElement | HTMLImageElement | SVGImageElement | HTMLVideoE
}; };
// проблема функции в том, что она не подходит для ссылок, пригодна только для blob'ов, потому что обычным ссылкам нужен 'load' каждый раз. // проблема функции в том, что она не подходит для ссылок, пригодна только для blob'ов, потому что обычным ссылкам нужен 'load' каждый раз.
export default function renderImageFromUrl(elem: HTMLElement | HTMLImageElement | SVGImageElement | HTMLVideoElement, url: string, callback?: (err?: Event) => void, useCache = true): boolean { export default async function renderImageFromUrl(elem: HTMLElement | HTMLImageElement | SVGImageElement | HTMLVideoElement, url: string, callback?: (err?: Event) => void, useCache = true) {
if(!url) { if(!url) {
console.error('renderImageFromUrl: no url?', elem, url); console.error('renderImageFromUrl: no url?', elem, url);
callback && callback(); callback && callback();
return false; return;
} }
if(((loadedURLs[url]/* && false */) && useCache) || elem instanceof HTMLVideoElement) { if(((loadedURLs[url]/* && false */) && useCache) || elem instanceof HTMLVideoElement) {
@ -25,7 +25,6 @@ export default function renderImageFromUrl(elem: HTMLElement | HTMLImageElement
} }
callback && callback(); callback && callback();
return true;
} else { } else {
const isImage = elem instanceof HTMLImageElement; const isImage = elem instanceof HTMLImageElement;
const loader = isImage ? elem as HTMLImageElement : new Image(); const loader = isImage ? elem as HTMLImageElement : new Image();
@ -53,7 +52,5 @@ export default function renderImageFromUrl(elem: HTMLElement | HTMLImageElement
if(callback) { if(callback) {
loader.addEventListener('error', callback); loader.addEventListener('error', callback);
} }
return false;
} }
} }

View File

@ -512,9 +512,10 @@ export class AppProfileManager {
return {cached, loadPromise: getAvatarPromise}; return {cached, loadPromise: getAvatarPromise};
} }
public putAvatar(div: HTMLElement, peerId: number, photo: UserProfilePhoto.userProfilePhoto | ChatPhoto.chatPhoto, size: PeerPhotoSize, img = new Image()) { public putAvatar(div: HTMLElement, peerId: number, photo: UserProfilePhoto.userProfilePhoto | ChatPhoto.chatPhoto, size: PeerPhotoSize, img = new Image(), onlyThumb = false) {
const {cached, loadPromise} = this.loadAvatar(peerId, photo, size); let {cached, loadPromise} = this.loadAvatar(peerId, photo, size);
let renderThumbPromise: Promise<void>;
let callback: () => void; let callback: () => void;
if(cached) { if(cached) {
// смотри в misc.ts: renderImageFromUrl // смотри в misc.ts: renderImageFromUrl
@ -528,41 +529,52 @@ export class AppProfileManager {
img.classList.add('fade-in'); img.classList.add('fade-in');
} }
let thumbImage: HTMLImageElement;
if(photo.stripped_thumb) {
thumbImage = new Image();
div.classList.add('avatar-relative');
thumbImage.classList.add('avatar-photo', 'avatar-photo-thumbnail');
img.classList.add('avatar-photo');
const url = appPhotosManager.getPreviewURLFromBytes(photo.stripped_thumb);
renderThumbPromise = renderImageFromUrl(thumbImage, url).then(() => {
replaceContent(div, thumbImage);
});
}
callback = () => { callback = () => {
if(photo.stripped_thumb) {
div.append(img);
} else {
replaceContent(div, img); replaceContent(div, img);
}
setTimeout(() => { setTimeout(() => {
if(div.childElementCount) { if(div.childElementCount) {
sequentialDom.mutateElement(img, () => {
div.dataset.color = ''; div.dataset.color = '';
if(animate) { if(animate) {
sequentialDom.mutateElement(img, () => {
img.classList.remove('fade-in'); img.classList.remove('fade-in');
});
} }
if(thumbImage) {
thumbImage.remove();
}
});
} }
}, animate ? 200 : 0); }, animate ? 200 : 0);
}; };
} }
const renderPromise = loadPromise.then((url) => { const renderPromise = loadPromise
/* if(photo.stripped_thumb) { .then((url) => renderImageFromUrl(img, url/* , false */))
url = appPhotosManager.getPreviewURLFromBytes(photo.stripped_thumb); .then(() => callback());
} */
return new Promise<void>((resolve) => { return {cached, loadPromise: renderThumbPromise || renderPromise};
renderImageFromUrl(img, url, () => {
callback();
resolve();
}/* , false */);
});
});
return {cached, loadPromise: renderPromise};
} }
// peerId === peerId || title // peerId === peerId || title
public putPhoto(div: HTMLElement, peerId: number, isDialog = false, title = '') { public putPhoto(div: HTMLElement, peerId: number, isDialog = false, title = '', onlyThumb = false) {
const photo = appPeersManager.getPeerPhoto(peerId); const photo = appPeersManager.getPeerPhoto(peerId);
const size: PeerPhotoSize = 'photo_small'; const size: PeerPhotoSize = 'photo_small';
@ -614,7 +626,7 @@ export class AppProfileManager {
} }
if(avatarAvailable/* && false */) { if(avatarAvailable/* && false */) {
return this.putAvatar(div, peerId, photo, size); return this.putAvatar(div, peerId, photo, size, undefined, onlyThumb);
} }
} }
} }

View File

@ -124,6 +124,10 @@ avatar-element {
cursor: pointer; cursor: pointer;
} }
&.avatar-relative {
position: relative;
}
/* &.avatar-54 { /* &.avatar-54 {
width: 54px; width: 54px;
height: 54px; height: 54px;
@ -198,3 +202,9 @@ avatar-element {
--multiplier: 3.375; --multiplier: 3.375;
} }
} }
.avatar-photo {
position: absolute;
top: 0;
left: 0;
}