fchat-rising/chat/preview/ImagePreview.vue

820 lines
26 KiB
Vue
Raw Normal View History

2019-06-08 02:26:01 +00:00
<template>
<!-- hiding elements instead of using 'v-if' is used here as an optimization -->
2020-04-01 22:19:55 +00:00
<div class="image-preview-wrapper" :class="{interactive: sticky, visible: visible}">
2020-04-01 19:23:23 +00:00
<div class="image-preview-toolbar" v-show="sticky || debug">
2019-07-04 19:34:21 +00:00
<a @click="toggleDevMode()" :class="{toggled: debug}" title="Debug Mode"><i class="fa fa-terminal"></i></a>
<a @click="toggleJsMode()" :class="{toggled: runJs}" title="Expand Images"><i class="fa fa-magic"></i></a>
<a @click="reloadUrl()" title="Reload Image"><i class="fa fa-redo-alt"></i></a>
2020-03-15 02:31:28 +00:00
<a @click="reset()" title="Reset Image Viewer"><i class="fa fa-recycle"></i></a>
2019-07-04 19:34:21 +00:00
<a @click="toggleStickyMode()" :class="{toggled: sticky}" title="Toggle Stickyness"><i class="fa fa-thumbtack"></i></a>
</div>
2020-03-15 18:17:36 +00:00
<!-- note: preload requires a webpack config CopyPlugin configuration -->
2020-04-01 19:23:23 +00:00
<webview
preload="./preview/assets/browser.pre.js"
src="about:blank"
webpreferences="autoplayPolicy=no-user-gesture-required,contextIsolation,sandbox,disableDialogs,disableHtmlFullScreenWindowResize,webSecurity,enableWebSQL=no,nodeIntegration=no,nativeWindowOpen=no,nodeIntegrationInWorker=no,nodeIntegrationInSubFrames=no,webviewTag=no"
2020-06-28 21:09:18 +00:00
enableremotemodule="false"
allowpopups="false"
nodeIntegration="false"
2021-09-11 20:43:33 +00:00
partition="persist:adblocked"
2020-06-28 21:09:18 +00:00
2020-04-01 19:23:23 +00:00
id="image-preview-ext"
ref="imagePreviewExt"
class="image-preview-external"
2020-10-25 22:55:21 +00:00
:style="previewStyles.ExternalImagePreviewHelper">
2020-04-01 19:23:23 +00:00
</webview>
2019-06-08 02:26:01 +00:00
<div
class="image-preview-local"
2020-10-25 22:55:21 +00:00
:style="previewStyles.LocalImagePreviewHelper"
2019-06-08 02:26:01 +00:00
>
</div>
2020-04-04 20:22:42 +00:00
2020-10-25 22:55:21 +00:00
<character-preview
:style="previewStyles.CharacterPreviewHelper"
ref="characterPreview"
></character-preview>
2020-04-04 20:22:42 +00:00
<i id="preview-spinner" class="fas fa-circle-notch fa-spin" v-show="shouldShowSpinner"></i>
<i id="preview-error" class="fas fa-times" v-show="shouldShowError"></i>
2019-06-08 02:26:01 +00:00
</div>
</template>
<script lang="ts">
2019-07-12 22:11:55 +00:00
import * as _ from 'lodash';
2019-06-08 02:26:01 +00:00
import {Component, Hook} from '@f-list/vue-ts';
import Vue from 'vue';
2020-03-15 16:23:39 +00:00
import core from '../core';
2019-07-06 16:49:19 +00:00
import { EventBus, EventBusEvent } from './event-bus';
2020-03-15 16:23:39 +00:00
import {domain} from '../../bbcode/core';
2020-04-05 18:29:43 +00:00
import {ImageDomMutator} from './image-dom-mutator';
2020-03-15 16:23:39 +00:00
2020-10-25 22:55:21 +00:00
import {
ExternalImagePreviewHelper,
LocalImagePreviewHelper,
PreviewManager,
CharacterPreviewHelper, RenderStyle
} from './helper';
2020-03-15 16:23:39 +00:00
2022-03-19 22:22:26 +00:00
import {Point} from 'electron';
2021-09-11 01:59:05 +00:00
import * as remote from '@electron/remote';
2019-07-06 16:49:19 +00:00
import Timer = NodeJS.Timer;
2020-03-15 02:31:28 +00:00
import IpcMessageEvent = Electron.IpcMessageEvent;
2020-10-25 22:55:21 +00:00
import CharacterPreview from './CharacterPreview.vue';
2019-07-06 16:49:19 +00:00
2020-04-01 22:19:55 +00:00
const screen = remote.screen;
2019-07-06 16:49:19 +00:00
2020-10-30 22:47:33 +00:00
const FLIST_PROFILE_MATCH = _.cloneDeep(/https?:\/\/(www.)?f-list.net\/c\/([a-zA-Z0-9+%_.!~*'()]+)\/?/);
2019-07-06 16:49:19 +00:00
interface DidFailLoadEvent extends Event {
errorCode: number;
errorDescription: string;
}
interface DidNavigateEvent extends Event {
httpResponseCode: number;
httpStatusText: string;
}
2020-10-25 22:55:21 +00:00
@Component({
components: {
'character-preview': CharacterPreview
}
})
2019-06-08 02:26:01 +00:00
export default class ImagePreview extends Vue {
2019-07-08 23:27:14 +00:00
private readonly MinTimePreviewVisible = 100;
2019-07-06 16:49:19 +00:00
visible = false;
2020-10-25 22:55:21 +00:00
previewManager = new PreviewManager(
this,
[
new ExternalImagePreviewHelper(this),
new LocalImagePreviewHelper(this),
new CharacterPreviewHelper(this)
// new ChannelPreviewHelper(this)
]
);
// externalPreviewHelper = new ExternalImagePreviewHelper(this);
// localPreviewHelper = new LocalImagePreviewHelper(this);
// externalPreviewStyle: Record<string, any> = {};
// localPreviewStyle: Record<string, any> = {};
2019-07-12 22:11:55 +00:00
2019-07-06 16:49:19 +00:00
url: string | null = null;
domain: string | undefined;
2019-07-06 16:49:19 +00:00
sticky = false;
runJs = true;
debug = false;
2019-06-08 02:26:01 +00:00
2020-04-05 18:29:43 +00:00
jsMutator = new ImageDomMutator(this.debug);
2019-10-06 23:08:22 +00:00
2020-04-04 20:22:42 +00:00
state = 'hidden';
shouldShowSpinner = false;
shouldShowError = true;
2019-07-06 16:49:19 +00:00
private interval: Timer | null = null;
2019-06-08 02:26:01 +00:00
2019-07-06 16:49:19 +00:00
private exitInterval: Timer | null = null;
private exitUrl: string | null = null;
private initialCursorPosition: Point | null = null;
private shouldDismiss = false;
private visibleSince = 0;
2020-10-25 22:55:21 +00:00
previewStyles: Record<string, RenderStyle> = {};
2019-06-08 02:26:01 +00:00
@Hook('mounted')
async onMounted(): Promise<void> {
2020-10-31 22:21:47 +00:00
console.info('Mounted ImagePreview');
2020-04-01 19:23:23 +00:00
2020-04-05 18:29:43 +00:00
// tslint:disable-next-line:no-floating-promises
this.jsMutator.init();
2019-06-08 02:26:01 +00:00
EventBus.$on(
'imagepreview-dismiss',
2019-07-06 16:49:19 +00:00
(eventData: EventBusEvent) => {
// console.log('Event dismiss', eventData.url);
2020-10-30 22:47:33 +00:00
this.dismiss(this.negotiateUrl(eventData.url as string || ''), eventData.force as boolean);
2019-06-08 02:26:01 +00:00
}
);
EventBus.$on(
'imagepreview-show',
2019-07-06 16:49:19 +00:00
(eventData: EventBusEvent) => {
// console.log('Event show', eventData.url);
2019-07-04 19:34:21 +00:00
2020-10-30 22:47:33 +00:00
const url = this.negotiateUrl(eventData.url as string || '');
const isInternalPreview = CharacterPreviewHelper.FLIST_CHARACTER_PROTOCOL_TESTER.test(url);
if (
((!core.state.settings.risingCharacterPreview) && (isInternalPreview))
|| ((!core.state.settings.risingLinkPreview) && (!isInternalPreview))
) {
return;
}
2020-10-30 22:47:33 +00:00
this.show(url);
2019-06-08 02:26:01 +00:00
}
);
2019-07-04 19:34:21 +00:00
EventBus.$on(
'imagepreview-toggle-stickyness',
2019-07-06 16:49:19 +00:00
(eventData: EventBusEvent) => {
if (!core.state.settings.risingLinkPreview) {
return;
}
2020-10-30 22:47:33 +00:00
const eventUrl = this.jsMutator.mutateUrl(this.negotiateUrl(eventData.url as string || ''));
2021-03-25 21:54:10 +00:00
if (
((eventData.force === true) || (this.url === eventUrl))
&& (this.visible)
) {
2019-07-04 19:34:21 +00:00
this.sticky = !this.sticky;
2021-03-25 21:54:10 +00:00
if (eventData.force) {
this.hide();
}
}
2019-07-04 19:34:21 +00:00
}
);
2019-07-07 01:37:15 +00:00
const webview = this.getWebview();
// clear preview cache, particularly cookies
// setInterval(
// () => remote.webContents.fromId(webview.getWebContentsId()).session.clearStorageData({storages: ['cookies', 'indexdb']}),
// 5000
// );
webview.addEventListener(
2019-10-06 23:08:22 +00:00
'update-target-url', // 'did-navigate', // 'dom-ready',
2019-07-06 16:49:19 +00:00
(event: EventBusEvent) => {
const url = webview.getURL();
2019-10-06 23:08:22 +00:00
const js = this.jsMutator.getMutatorJsForSite(url, 'update-target-url');
2020-03-30 21:52:25 +00:00
// tslint:disable-next-line
this.executeJavaScript(js, 'update-target-url', event);
2019-07-04 19:34:21 +00:00
}
);
2019-07-12 22:11:55 +00:00
2019-10-06 23:08:22 +00:00
webview.addEventListener(
'dom-ready', // 'did-navigate', // 'dom-ready',
(event: EventBusEvent) => {
const url = webview.getURL();
const js = this.jsMutator.getMutatorJsForSite(url, 'dom-ready');
2020-03-30 21:52:25 +00:00
// tslint:disable-next-line
this.executeJavaScript(js, 'dom-ready', event);
2020-04-04 20:22:42 +00:00
this.setState('loaded');
2019-10-06 23:08:22 +00:00
}
);
2019-07-04 19:34:21 +00:00
webview.addEventListener(
'did-fail-load',
2019-07-06 16:49:19 +00:00
(event: Event) => {
2020-04-04 20:22:42 +00:00
2019-07-06 16:49:19 +00:00
const e = event as DidFailLoadEvent;
2019-07-04 19:34:21 +00:00
2020-04-04 20:22:42 +00:00
if (e.errorCode !== -3) {
this.setState('error'); // -3 is a weird error code, not sure why it occurs
}
2020-03-30 21:52:25 +00:00
if (e.errorCode < 0) {
2020-04-04 18:38:00 +00:00
const url = webview.getURL();
if (url.match(/^https?:\/\/(www.)?pornhub.com/)) {
const qjs = this.jsMutator.getMutatorJsForSite(url, 'update-target-url')
|| this.jsMutator.getMutatorJsForSite(url, 'dom-ready');
// tslint:disable-next-line
this.executeJavaScript(qjs, 'did-fail-load-but-still-loading', event);
return;
}
// console.error('DID FAIL LOAD', event);
// const url = this.getUrl() || '';
//
// const qjs = this.jsMutator.getMutatorJsForSite(url, 'update-target-url')
// || this.jsMutator.getMutatorJsForSite(url, 'dom-ready');
//
// // tslint:disable-next-line
// this.executeJavaScript(qjs, 'did-fail-load-but-still-loading', event);
return;
2020-03-30 21:52:25 +00:00
}
2019-07-06 16:49:19 +00:00
2020-03-30 21:52:25 +00:00
// if (e.errorCode < 100) {
// const url = webview.getURL();
// const js = this.jsMutator.getMutatorJsForSite(url, 'update-target-url');
//
// this.executeJavaScript(js, 'did-fail-load-but-still-loading', event);
//
// return;
// }
2019-07-04 19:34:21 +00:00
2020-03-30 21:52:25 +00:00
const js = this.jsMutator.getErrorMutator(e.errorCode, e.errorDescription);
// tslint:disable-next-line
this.executeJavaScript(js, 'did-fail-load', event);
}
);
2019-07-04 19:34:21 +00:00
webview.addEventListener(
'did-navigate',
2019-07-06 16:49:19 +00:00
(event: Event) => {
const e = event as DidNavigateEvent;
if (e.httpResponseCode >= 400) {
const js = this.jsMutator.getErrorMutator(e.httpResponseCode, e.httpStatusText);
2019-07-04 19:34:21 +00:00
2020-03-30 21:52:25 +00:00
// tslint:disable-next-line
this.executeJavaScript(js, 'did-navigate', event);
2019-07-04 19:34:21 +00:00
}
}
);
// webview.getWebContents().on(
webview.addEventListener(
'did-finish-load',
2019-07-06 16:49:19 +00:00
(event: Event) => {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview did-finish-load', event);
}
);
2019-07-04 19:34:21 +00:00
2020-03-15 02:31:28 +00:00
webview.addEventListener(
'ipc-message',
(event: IpcMessageEvent) => {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview ipc-message', event);
2020-03-15 02:31:28 +00:00
if (event.channel === 'webview.img') {
2020-03-15 14:02:31 +00:00
// tslint:disable-next-line:no-unsafe-any
this.updatePreviewSize(parseInt(event.args[0], 10), parseInt(event.args[1], 10));
2020-03-15 02:31:28 +00:00
}
}
);
2020-04-04 20:22:42 +00:00
// const webContentsId = webview.getWebContentsId();
//
// remote.webContents.fromId(webContentsId).session.on(
// 'will-download',
// (e: Event) => {
// e.preventDefault();
// }
// );
2019-10-27 19:31:45 +00:00
2019-07-12 22:11:55 +00:00
_.each(
2020-03-15 02:31:28 +00:00
['did-start-loading', 'load-commit', 'dom-ready', 'will-navigate', 'did-navigate', 'did-navigate-in-page', 'update-target-url', 'ipc-message'],
2019-07-12 22:11:55 +00:00
(en: string) => {
webview.addEventListener(
en,
(event: Event) => {
2020-03-30 21:52:25 +00:00
this.debugLog(`ImagePreview ${en} ${Date.now()}`, event);
2019-07-12 22:11:55 +00:00
}
);
}
);
setInterval(
() => {
if (((this.visible) && (!this.exitInterval) && (!this.shouldDismiss)) || (this.interval))
this.initialCursorPosition = screen.getCursorScreenPoint();
2019-07-12 22:11:55 +00:00
if ((this.visible) && (this.shouldDismiss) && (this.hasMouseMovedSince()) && (!this.exitInterval) && (!this.interval)) {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: call hide from interval');
2019-07-12 22:11:55 +00:00
this.hide();
2019-07-12 22:11:55 +00:00
}
2020-04-04 20:22:42 +00:00
this.shouldShowSpinner = this.testSpinner();
this.shouldShowError = this.testError();
},
2020-03-15 14:02:31 +00:00
50
);
2019-06-08 02:26:01 +00:00
}
2020-03-15 02:31:28 +00:00
reRenderStyles(): void {
2020-10-25 22:55:21 +00:00
this.previewStyles = this.previewManager.renderStyles();
2020-03-15 02:31:28 +00:00
}
2020-10-30 22:47:33 +00:00
negotiateUrl(url: string): string {
const match = url.match(FLIST_PROFILE_MATCH);
if (!match) {
return url;
}
return `flist-character://${decodeURI(match[2])}`;
}
2020-03-15 02:31:28 +00:00
updatePreviewSize(width: number, height: number): void {
2020-10-25 22:55:21 +00:00
const helper = this.previewManager.getVisiblePreview();
if ((!helper) || (!helper.reactsToSizeUpdates())) {
return;
2020-03-15 02:31:28 +00:00
}
if ((width) && (height)) {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: updatePreviewSize', width, height, width / height);
2020-03-15 02:31:28 +00:00
2020-10-25 22:55:21 +00:00
helper.setRatio(width / height);
2020-03-15 02:31:28 +00:00
this.reRenderStyles();
}
}
2019-07-04 19:34:21 +00:00
hide(): void {
this.cancelExitTimer();
this.url = null;
this.visible = false;
2020-10-25 22:55:21 +00:00
this.previewManager.hide();
this.exitUrl = null;
this.exitInterval = null;
this.shouldDismiss = false;
2019-07-04 19:34:21 +00:00
this.sticky = false;
2020-03-15 02:31:28 +00:00
2020-04-04 20:22:42 +00:00
this.setState('hidden');
2020-03-15 02:31:28 +00:00
this.reRenderStyles();
}
dismiss(initialUrl: string, force: boolean = false): void {
const url = this.jsMutator.mutateUrl(initialUrl);
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: dismiss', url);
2019-07-12 22:11:55 +00:00
if (this.url !== url)
return; // simply ignore
2019-07-04 19:34:21 +00:00
// if (this.debug)
// return;
if (this.sticky)
2019-06-24 01:02:40 +00:00
return;
// console.log('DISMISS');
const due = this.visible ? this.MinTimePreviewVisible - Math.min(this.MinTimePreviewVisible, (Date.now() - this.visibleSince)) : 0;
2019-06-08 02:26:01 +00:00
this.cancelTimer();
if (this.exitInterval)
return;
this.exitUrl = this.url;
this.shouldDismiss = true;
2019-07-12 22:11:55 +00:00
if ((!this.hasMouseMovedSince()) && (!force))
return;
2020-10-30 22:47:33 +00:00
this.debugLog('ImagePreview: dismiss.exec', due, this.previewManager.getVisibilityStatus(), url);
2019-07-12 22:11:55 +00:00
// This timeout makes the preview window disappear with a slight delay, which helps UX
// when dealing with situations such as quickly scrolling text that moves the cursor away
// from the link
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line no-unnecessary-type-assertion
this.exitInterval = setTimeout(
() => this.hide(),
due
2019-07-06 16:49:19 +00:00
) as Timer;
2019-06-08 02:26:01 +00:00
}
show(initialUrl: string): void {
const url = this.jsMutator.mutateUrl(initialUrl);
2020-10-25 22:55:21 +00:00
this.debugLog('ImagePreview: show', this.previewManager.getVisibilityStatus(),
2019-07-12 22:11:55 +00:00
this.visible, this.hasMouseMovedSince(), !!this.interval, this.sticky, url);
// console.log('SHOW');
2019-07-12 22:11:55 +00:00
if ((this.visible) && (!this.exitInterval) && (!this.hasMouseMovedSince())) {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: show cancel: visible & not moved');
return;
2019-07-12 22:11:55 +00:00
}
if ((this.url === url) && ((this.visible) || (this.interval))) {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: same url', url, this.url);
return;
2019-07-12 22:11:55 +00:00
}
if ((this.url) && (this.sticky) && (this.visible)) {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: sticky visible');
2019-07-04 19:34:21 +00:00
return;
2019-07-12 22:11:55 +00:00
}
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: show.exec', url);
2019-07-04 19:34:21 +00:00
const due = ((url === this.exitUrl) && (this.exitInterval)) ? 0 : 200;
2019-06-08 02:26:01 +00:00
this.url = url;
this.domain = domain(url);
this.cancelExitTimer();
2019-06-08 02:26:01 +00:00
this.cancelTimer();
// This timer makes sure that just by accidentally brushing across a link won't show (blink) the preview
// -- you actually have to pause on it
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line no-unnecessary-type-assertion
2019-06-08 02:26:01 +00:00
this.interval = setTimeout(
2020-03-15 14:02:31 +00:00
() => {
2020-03-30 21:52:25 +00:00
this.debugLog('ImagePreview: show.timeout', this.url);
2019-07-12 22:11:55 +00:00
2020-10-25 22:55:21 +00:00
const helper = this.previewManager.show(this.url || undefined, this.domain);
2020-03-15 14:02:31 +00:00
this.interval = null;
2019-06-08 02:26:01 +00:00
this.visible = true;
this.visibleSince = Date.now();
this.shouldDismiss = false;
this.initialCursorPosition = screen.getCursorScreenPoint();
2020-03-15 02:31:28 +00:00
this.reRenderStyles();
2020-04-04 20:22:42 +00:00
2020-10-25 22:55:21 +00:00
if (helper) {
this.setState(helper.shouldTrackLoading() ? 'loading' : 'loaded');
} else {
this.setState('loaded');
}
2019-06-08 02:26:01 +00:00
},
due
2019-07-06 16:49:19 +00:00
) as Timer;
2019-06-08 02:26:01 +00:00
}
hasMouseMovedSince(): boolean {
if (!this.initialCursorPosition)
return true;
2019-06-08 02:26:01 +00:00
try {
const p = screen.getCursorScreenPoint();
return ((p.x !== this.initialCursorPosition.x) || (p.y !== this.initialCursorPosition.y));
} catch (err) {
2019-07-12 22:11:55 +00:00
console.error('ImagePreview', err);
return true;
2019-06-08 02:26:01 +00:00
}
}
cancelTimer(): void {
if (this.interval)
clearTimeout(this.interval);
2019-06-08 02:26:01 +00:00
this.interval = null;
}
cancelExitTimer(): void {
if (this.exitInterval)
clearTimeout(this.exitInterval);
this.exitInterval = null;
}
isVisible(): boolean {
2019-06-08 02:26:01 +00:00
return this.visible;
}
getUrl(): string | null {
2019-06-08 02:26:01 +00:00
return this.url;
}
2019-10-06 23:08:22 +00:00
/* isExternalUrl(): boolean {
2019-07-04 19:34:21 +00:00
// 'f-list.net' is tested here on purpose, because keeps the character URLs from being previewed
2019-06-08 02:26:01 +00:00
return !((this.domain === 'f-list.net') || (this.domain === 'static.f-list.net'));
}
isInternalUrl(): boolean {
2019-06-08 02:26:01 +00:00
return !this.isExternalUrl();
2019-10-06 23:08:22 +00:00
}*/
2019-07-04 19:34:21 +00:00
toggleDevMode(): void {
this.debug = !this.debug;
2019-07-06 16:49:19 +00:00
this.jsMutator.setDebug(this.debug);
2020-10-25 22:55:21 +00:00
this.previewManager.setDebug(this.debug);
2019-07-06 16:49:19 +00:00
2019-07-04 19:34:21 +00:00
if (this.debug) {
2019-07-07 01:37:15 +00:00
const webview = this.getWebview();
2019-07-04 19:34:21 +00:00
webview.openDevTools();
}
}
2020-03-30 21:52:25 +00:00
async executeJavaScript(js: string | undefined, context: string = 'unknown', logDetails?: any): Promise<any> {
2023-12-03 00:41:17 +00:00
console.log('EXECUTE JS', js);
if (!this.runJs) return;
2020-03-30 21:52:25 +00:00
const webview = this.getWebview();
if (!js) {
this.debugLog(`ImagePreview ${context}: No JavaScript to execute`, logDetails);
return;
}
2020-04-01 22:19:55 +00:00
this.debugLog(`ImagePreview execute-${context}`, js, logDetails);
2020-03-30 21:52:25 +00:00
2020-04-01 22:19:55 +00:00
try {
const result = await (webview.executeJavaScript(js) as unknown as Promise<any>);
2020-03-30 21:52:25 +00:00
2020-04-01 22:19:55 +00:00
this.debugLog(`ImagePreview result-${context}`, result);
2020-03-30 21:52:25 +00:00
2020-04-01 22:19:55 +00:00
return result;
} catch (err) {
this.debugLog(`ImagePreview error-${context}`, err);
}
2020-03-30 21:52:25 +00:00
}
debugLog(...args: any[]): void {
if (this.debug) {
console.log(...args);
}
}
2019-07-04 19:34:21 +00:00
toggleStickyMode(): void {
this.sticky = !this.sticky;
if (!this.sticky)
this.hide();
}
2020-10-25 22:55:21 +00:00
2019-07-04 19:34:21 +00:00
toggleJsMode(): void {
this.runJs = !this.runJs;
}
2020-10-25 22:55:21 +00:00
2019-07-04 19:34:21 +00:00
reloadUrl(): void {
2020-10-25 22:55:21 +00:00
const helper = this.previewManager.getVisiblePreview();
2019-07-04 19:34:21 +00:00
2020-10-25 22:55:21 +00:00
if ((!helper) || (!helper.usesWebView())) {
return;
2019-07-04 19:34:21 +00:00
}
2020-10-25 22:55:21 +00:00
// helper.reload();
this.getWebview().reload();
2019-07-04 19:34:21 +00:00
}
2019-07-07 01:37:15 +00:00
2020-10-25 22:55:21 +00:00
2022-03-19 22:22:26 +00:00
getWebview(): Electron.WebviewTag {
return this.$refs.imagePreviewExt as Electron.WebviewTag;
2019-07-07 01:37:15 +00:00
}
2020-03-15 02:31:28 +00:00
2020-04-01 22:19:55 +00:00
2020-10-25 22:55:21 +00:00
getCharacterPreview(): CharacterPreview {
return this.$refs.characterPreview as CharacterPreview;
}
2020-03-15 14:02:31 +00:00
reset(): void {
2020-10-25 22:55:21 +00:00
this.previewManager = new PreviewManager(
this,
[
new ExternalImagePreviewHelper(this),
new LocalImagePreviewHelper(this),
new CharacterPreviewHelper(this)
// new ChannelPreviewHelper(this)
]
);
2020-03-15 02:31:28 +00:00
this.url = null;
this.domain = undefined;
this.sticky = false;
this.runJs = true;
this.debug = false;
2020-04-05 18:29:43 +00:00
this.jsMutator = new ImageDomMutator(this.debug);
2020-03-15 02:31:28 +00:00
this.cancelExitTimer();
this.cancelTimer();
this.exitUrl = null;
this.initialCursorPosition = null;
this.shouldDismiss = false;
this.visibleSince = 0;
2020-04-04 20:22:42 +00:00
this.shouldShowSpinner = false;
this.shouldShowError = false;
this.setState('hidden');
2020-03-15 02:31:28 +00:00
this.reRenderStyles();
}
2020-04-04 20:22:42 +00:00
setState(state: string): void {
this.debugLog('ImagePreview set-state', state, (this.visibleSince > 0) ? `${(Date.now() - this.visibleSince) / 1000}s` : '');
this.state = state;
this.shouldShowSpinner = this.testSpinner();
this.shouldShowError = this.testError();
}
testSpinner(): boolean {
return (this.visibleSince > 0)
? ((this.state === 'loading') && (Date.now() - this.visibleSince > 1000))
: false;
}
2020-10-25 22:55:21 +00:00
2020-04-04 20:22:42 +00:00
testError(): boolean {
2020-10-25 22:55:21 +00:00
const helper = this.previewManager.getVisiblePreview();
if ((!helper) || (!helper.usesWebView())) {
return false;
}
return (this.state === 'error');
2020-04-04 20:22:42 +00:00
}
2019-06-08 02:26:01 +00:00
}
</script>
<style lang="scss">
2020-03-15 16:23:39 +00:00
@import "../../node_modules/bootstrap/scss/functions";
@import "../../node_modules/bootstrap/scss/variables";
@import "../../node_modules/bootstrap/scss/mixins/breakpoints";
2019-06-08 02:26:01 +00:00
2019-07-04 19:34:21 +00:00
.image-preview-wrapper {
z-index: 10000;
display: none;
2019-06-08 02:26:01 +00:00
position: absolute;
left: 0;
2019-07-04 19:34:21 +00:00
top: 0;
2019-06-08 02:26:01 +00:00
width: 50%;
2019-06-23 21:48:41 +00:00
height: 70%;
2019-06-08 02:26:01 +00:00
pointer-events: none;
2020-03-15 18:17:36 +00:00
overflow: visible;
2019-07-04 19:34:21 +00:00
&.visible {
display: block;
}
&.interactive {
pointer-events: auto;
.image-preview-local,
.image-preview-auto {
// pointer-events: auto;
}
}
.image-preview-external {
/* position: absolute;
width: 50%;
height: 70%;
top: 0;
left: 0; */
width: 100%;
height: 100%;
// pointer-events: none;
background-color: black;
}
.image-preview-local {
/* position: absolute;
width: 50%;
height: 70%;
top: 0;
left: 0; */
width: 100%;
height: 100%;
// pointer-events: none;
background-size: contain;
background-position: top left;
background-repeat: no-repeat;
// background-color: black;
}
.image-preview-toolbar {
position: absolute;
/* background-color: green; */
left: 0;
top: 0;
margin: 1rem;
height: 3.5rem;
display: flex;
-webkit-backdrop-filter: blur(10px);
flex-direction: row;
width: 15rem;
flex-wrap: nowrap;
background-color: rgba(77, 76, 116, 0.92);
border-radius: 3px;
border: 1px solid rgba(255, 255, 255, 0.3);
padding: 0.5rem;
box-shadow: 2px 2px 3px rgba(0, 0, 0, 0.2);
2020-10-25 22:55:21 +00:00
z-index: 1000;
2019-07-04 19:34:21 +00:00
a i.fa {
font-size: 1.25rem;
top: 50%;
position: relative;
transform: translateY(-50%);
}
a {
flex: 1;
text-align: center;
border: 1px solid rgba(255, 255, 255, 0.25);
border-radius: 3px;
margin-right: 0.5rem;
background-color: rgba(0, 0, 0, 0.1);
}
a:last-child {
margin-right: 0;
}
.toggled {
background-color: rgba(255, 255, 255, 0.2);
box-shadow: 0 0 1px 0px rgba(255, 255, 255, 0.6);
}
}
2020-04-04 20:22:42 +00:00
#preview-spinner {
color: white;
opacity: 0.5;
transition: visibility 0.25s, opacity 0.25s;
font-size: 30pt;
position: absolute;
left: 1rem;
top: 1rem;
transform: translateX(-50%), translateY(-50%);
text-shadow: 0 0 2px #b3b3b3;
}
#preview-error {
color: red;
transition: all 0.25s;
font-size: 180pt;
position: absolute;
left: 2rem;
top: 0;
opacity: 0.8;
}
}
2019-06-08 02:26:01 +00:00
</style>