mirror of
https://github.com/0PandaDEV/Qopy.git
synced 2025-04-21 21:24:05 +02:00
880 lines
26 KiB
Vue
880 lines
26 KiB
Vue
<template>
|
|
<div class="bg" tabindex="0">
|
|
<input
|
|
ref="searchInput"
|
|
v-model="searchQuery"
|
|
@input="searchHistory"
|
|
autocorrect="off"
|
|
autocapitalize="off"
|
|
spellcheck="false"
|
|
class="search"
|
|
type="text"
|
|
placeholder="Type to filter entries..." />
|
|
<div class="bottom-bar">
|
|
<div class="left">
|
|
<img class="logo" width="18px" src="../public/logo.png" alt="" />
|
|
<p>Qopy</p>
|
|
</div>
|
|
<div class="right">
|
|
<div class="paste" @click="pasteSelectedItem">
|
|
<p>Paste</p>
|
|
<img src="../public/enter.svg" alt="" />
|
|
</div>
|
|
<div class="divider"></div>
|
|
<div class="actions">
|
|
<p>Actions</p>
|
|
<div>
|
|
<img
|
|
v-if="os === 'windows' || os === 'linux'"
|
|
src="../public/ctrl.svg"
|
|
alt="" />
|
|
<img v-if="os === 'macos'" src="../public/cmd.svg" alt="" />
|
|
<img src="../public/k.svg" alt="" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<OverlayScrollbarsComponent
|
|
class="results"
|
|
ref="resultsContainer"
|
|
:options="{ scrollbars: { autoHide: 'scroll' } }">
|
|
<template v-for="(group, groupIndex) in groupedHistory" :key="groupIndex">
|
|
<div class="time-separator">{{ group.label }}</div>
|
|
<div
|
|
v-for="(item, index) in group.items"
|
|
:key="item.id"
|
|
:class="[
|
|
'result clothoid-corner',
|
|
{ selected: isSelected(groupIndex, index) },
|
|
]"
|
|
@click="selectItem(groupIndex, index)"
|
|
:ref="
|
|
(el: any) => {
|
|
if (isSelected(groupIndex, index))
|
|
selectedElement = el as HTMLElement;
|
|
}
|
|
">
|
|
<template v-if="item.content_type === 'image'">
|
|
<img
|
|
v-if="imageUrls[item.id]"
|
|
:src="imageUrls[item.id]"
|
|
alt="Image"
|
|
class="image"
|
|
@error="onImageError" />
|
|
<img v-else src="../public/icons/Image.svg" class="icon" />
|
|
</template>
|
|
<template v-else-if="hasFavicon(item.favicon ?? '')">
|
|
<img
|
|
:src="
|
|
item.favicon
|
|
? getFaviconFromDb(item.favicon)
|
|
: '../public/icons/Link.svg'
|
|
"
|
|
alt="Favicon"
|
|
class="favicon"
|
|
@error="
|
|
($event.target as HTMLImageElement).src =
|
|
'../public/icons/Link.svg'
|
|
" />
|
|
</template>
|
|
<img
|
|
src="../public/icons/File.svg"
|
|
class="icon"
|
|
v-else-if="item.content_type === ContentType.File" />
|
|
<img
|
|
src="../public/icons/Text.svg"
|
|
class="icon"
|
|
v-else-if="item.content_type === ContentType.Text" />
|
|
<div v-else-if="item.content_type === ContentType.Color">
|
|
<svg
|
|
width="18"
|
|
height="18"
|
|
viewBox="0 0 18 18"
|
|
fill="none"
|
|
xmlns="http://www.w3.org/2000/svg">
|
|
<g>
|
|
<rect width="18" height="18" />
|
|
<path
|
|
d="M9 18C12.2154 18 15.1865 16.2846 16.7942 13.5C18.4019 10.7154 18.4019 7.28461 16.7942 4.5C15.1865 1.71539 12.2154 -1.22615e-06 9 0C5.78461 0 2.81347 1.71539 1.20577 4.5C-0.401925 7.28461 -0.401923 10.7154 1.20577 13.5C2.81347 16.2846 5.78461 18 9 18Z"
|
|
fill="#E5DFD5" />
|
|
<path
|
|
d="M9 16C7.14348 16 5.36301 15.2625 4.05025 13.9497C2.7375 12.637 2 10.8565 2 9C2 7.14348 2.7375 5.36301 4.05025 4.05025C5.36301 2.7375 7.14348 2 9 2C10.8565 2 12.637 2.7375 13.9497 4.05025C15.2625 5.36301 16 7.14348 16 9C16 10.8565 15.2625 12.637 13.9497 13.9497C12.637 15.2625 10.8565 16 9 16Z"
|
|
:fill="item.content" />
|
|
</g>
|
|
</svg>
|
|
</div>
|
|
<img
|
|
src="../public/icons/Code.svg"
|
|
class="icon"
|
|
v-else-if="item.content_type === ContentType.Code" />
|
|
<span v-if="item.content_type === ContentType.Image">
|
|
Image ({{ imageDimensions[item.id] || "Loading..." }})
|
|
</span>
|
|
<span v-else>{{ truncateContent(item.content) }}</span>
|
|
</div>
|
|
</template>
|
|
</OverlayScrollbarsComponent>
|
|
|
|
<div
|
|
class="content"
|
|
v-if="selectedItem?.content_type === ContentType.Image">
|
|
<img :src="imageUrls[selectedItem.id]" alt="Image" class="image" />
|
|
</div>
|
|
<div
|
|
v-else-if="selectedItem && isYoutubeWatchUrl(selectedItem.content)"
|
|
class="content">
|
|
<img
|
|
class="image"
|
|
:src="getYoutubeThumbnail(selectedItem.content)"
|
|
alt="YouTube Thumbnail" />
|
|
</div>
|
|
<div
|
|
class="content"
|
|
v-else-if="
|
|
selectedItem?.content_type === ContentType.Link && pageOgImage
|
|
">
|
|
<img :src="pageOgImage" alt="Image" class="image" />
|
|
</div>
|
|
<OverlayScrollbarsComponent v-else class="content">
|
|
<span>{{ selectedItem?.content || "" }}</span>
|
|
</OverlayScrollbarsComponent>
|
|
|
|
<OverlayScrollbarsComponent
|
|
class="information"
|
|
:options="{ scrollbars: { autoHide: 'scroll' } }">
|
|
<div class="title">Information</div>
|
|
<div class="info-content" v-if="selectedItem && getInfo">
|
|
<div class="info-row" v-for="(row, index) in infoRows" :key="index">
|
|
<p class="label">{{ row.label }}</p>
|
|
<span :class="{ 'url-truncate': row.isUrl }" :data-text="row.value">
|
|
{{ row.value }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</OverlayScrollbarsComponent>
|
|
<Noise />
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, computed, onMounted, watch, nextTick, shallowRef } from "vue";
|
|
import { OverlayScrollbarsComponent } from "overlayscrollbars-vue";
|
|
import "overlayscrollbars/overlayscrollbars.css";
|
|
import { app, window } from "@tauri-apps/api";
|
|
import { platform } from "@tauri-apps/plugin-os";
|
|
import { listen } from "@tauri-apps/api/event";
|
|
import { useNuxtApp } from "#app";
|
|
import { invoke } from "@tauri-apps/api/core";
|
|
import { HistoryItem, ContentType } from "~/types/types";
|
|
import type {
|
|
InfoText,
|
|
InfoImage,
|
|
InfoFile,
|
|
InfoLink,
|
|
InfoColor,
|
|
InfoCode,
|
|
} from "~/types/types";
|
|
import { Key } from "wrdu-keyboard/key";
|
|
|
|
interface GroupedHistory {
|
|
label: string;
|
|
items: HistoryItem[];
|
|
}
|
|
|
|
const { $history } = useNuxtApp();
|
|
const CHUNK_SIZE = 50;
|
|
const SCROLL_THRESHOLD = 100;
|
|
|
|
const history = shallowRef<HistoryItem[]>([]);
|
|
let offset = 0;
|
|
let isLoading = false;
|
|
|
|
const resultsContainer = shallowRef<InstanceType<
|
|
typeof OverlayScrollbarsComponent
|
|
> | null>(null);
|
|
const searchQuery = ref("");
|
|
const selectedGroupIndex = ref(0);
|
|
const selectedItemIndex = ref(0);
|
|
const selectedElement = shallowRef<HTMLElement | null>(null);
|
|
const searchInput = ref<HTMLInputElement | null>(null);
|
|
const os = ref<string>("");
|
|
const imageUrls = shallowRef<Record<string, string>>({});
|
|
const imageDimensions = shallowRef<Record<string, string>>({});
|
|
const imageSizes = shallowRef<Record<string, string>>({});
|
|
const lastUpdateTime = ref<number>(Date.now());
|
|
const imageLoadError = ref<boolean>(false);
|
|
const imageLoading = ref<boolean>(false);
|
|
const pageTitle = ref<string>("");
|
|
const pageOgImage = ref<string>("");
|
|
|
|
const keyboard = useKeyboard();
|
|
|
|
const isSameDay = (date1: Date, date2: Date): boolean => {
|
|
return (
|
|
date1.getFullYear() === date2.getFullYear() &&
|
|
date1.getMonth() === date2.getMonth() &&
|
|
date1.getDate() === date2.getDate()
|
|
);
|
|
};
|
|
|
|
const getWeekNumber = (date: Date): number => {
|
|
const firstDayOfYear = new Date(date.getFullYear(), 0, 1);
|
|
return Math.ceil(
|
|
((date.getTime() - firstDayOfYear.getTime()) / 86400000 +
|
|
firstDayOfYear.getDay() +
|
|
1) /
|
|
7
|
|
);
|
|
};
|
|
|
|
const groupedHistory = computed<GroupedHistory[]>(() => {
|
|
const now = new Date();
|
|
const today = new Date(now.getFullYear(), now.getMonth(), now.getDate());
|
|
const thisWeek = getWeekNumber(now);
|
|
const thisYear = now.getFullYear();
|
|
|
|
const groups: Record<string, HistoryItem[]> = {
|
|
Today: [],
|
|
Yesterday: [],
|
|
"This Week": [],
|
|
"Last Week": [],
|
|
"This Year": [],
|
|
"Last Year": [],
|
|
};
|
|
|
|
const filteredItems = searchQuery.value
|
|
? history.value.filter((item) =>
|
|
item.content.toLowerCase().includes(searchQuery.value.toLowerCase())
|
|
)
|
|
: history.value;
|
|
|
|
const yesterday = new Date(today.getTime() - 86400000);
|
|
|
|
filteredItems.forEach((item) => {
|
|
const itemDate = new Date(item.timestamp);
|
|
const itemWeek = getWeekNumber(itemDate);
|
|
const itemYear = itemDate.getFullYear();
|
|
|
|
if (isSameDay(itemDate, today)) groups.Today.push(item);
|
|
else if (isSameDay(itemDate, yesterday)) groups.Yesterday.push(item);
|
|
else if (itemYear === thisYear && itemWeek === thisWeek)
|
|
groups["This Week"].push(item);
|
|
else if (itemYear === thisYear && itemWeek === thisWeek - 1)
|
|
groups["Last Week"].push(item);
|
|
else if (itemYear === thisYear) groups["This Year"].push(item);
|
|
else groups["Last Year"].push(item);
|
|
});
|
|
|
|
return Object.entries(groups)
|
|
.filter(([_, items]) => items.length > 0)
|
|
.map(([label, items]) => ({ label, items }));
|
|
});
|
|
|
|
const selectedItem = computed<HistoryItem | null>(() => {
|
|
const group = groupedHistory.value[selectedGroupIndex.value];
|
|
return group?.items[selectedItemIndex.value] ?? null;
|
|
});
|
|
|
|
const loadHistoryChunk = async (): Promise<void> => {
|
|
if (isLoading) return;
|
|
isLoading = true;
|
|
|
|
try {
|
|
const results = await $history.loadHistoryChunk(offset, CHUNK_SIZE);
|
|
if (!results.length) {
|
|
isLoading = false;
|
|
return;
|
|
}
|
|
|
|
const processedItems = await Promise.all(
|
|
results.map(async (item) => {
|
|
const historyItem = new HistoryItem(
|
|
item.source,
|
|
item.content_type,
|
|
item.content,
|
|
item.favicon,
|
|
item.source_icon,
|
|
item.language
|
|
);
|
|
Object.assign(historyItem, {
|
|
id: item.id,
|
|
timestamp: new Date(item.timestamp),
|
|
});
|
|
|
|
if (historyItem.content_type === ContentType.Image) {
|
|
try {
|
|
const base64 = await $history.readImage({
|
|
filename: historyItem.content,
|
|
});
|
|
const size = Math.ceil((base64.length * 3) / 4);
|
|
imageSizes.value[historyItem.id] = formatFileSize(size);
|
|
|
|
const img = new Image();
|
|
img.src = `data:image/png;base64,${base64}`;
|
|
imageUrls.value[historyItem.id] = img.src;
|
|
|
|
await new Promise<void>((resolve) => {
|
|
img.onload = () => {
|
|
imageDimensions.value[
|
|
historyItem.id
|
|
] = `${img.width}x${img.height}`;
|
|
resolve();
|
|
};
|
|
img.onerror = () => {
|
|
imageDimensions.value[historyItem.id] = "Error";
|
|
resolve();
|
|
};
|
|
});
|
|
} catch (error) {
|
|
console.error("Error processing image:", error);
|
|
imageDimensions.value[historyItem.id] = "Error";
|
|
imageSizes.value[historyItem.id] = "Error";
|
|
}
|
|
}
|
|
return historyItem;
|
|
})
|
|
);
|
|
|
|
history.value = [...history.value, ...processedItems];
|
|
offset += CHUNK_SIZE;
|
|
} catch (error) {
|
|
console.error("Failed to load history:", error);
|
|
} finally {
|
|
isLoading = false;
|
|
}
|
|
};
|
|
|
|
const handleScroll = (): void => {
|
|
const viewport = resultsContainer.value?.osInstance()?.elements().viewport;
|
|
if (!viewport) return;
|
|
|
|
const { scrollTop, scrollHeight, clientHeight } = viewport;
|
|
if (scrollHeight - scrollTop - clientHeight < SCROLL_THRESHOLD) {
|
|
loadHistoryChunk();
|
|
}
|
|
};
|
|
|
|
const scrollToSelectedItem = (forceScrollTop: boolean = false): void => {
|
|
nextTick(() => {
|
|
const osInstance = resultsContainer.value?.osInstance();
|
|
const viewport = osInstance?.elements().viewport;
|
|
if (!selectedElement.value || !viewport) return;
|
|
|
|
if (!forceScrollTop) {
|
|
const viewportRect = viewport.getBoundingClientRect();
|
|
const elementRect = selectedElement.value.getBoundingClientRect();
|
|
|
|
const isAbove = elementRect.top < viewportRect.top;
|
|
const isBelow = elementRect.bottom > viewportRect.bottom - 8;
|
|
|
|
if (isAbove || isBelow) {
|
|
const scrollOffset = isAbove
|
|
? elementRect.top -
|
|
viewportRect.top -
|
|
(selectedItemIndex.value === 0 ? 36 : 8)
|
|
: elementRect.bottom - viewportRect.bottom + 9;
|
|
|
|
viewport.scrollBy({ top: scrollOffset, behavior: "smooth" });
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
const isSelected = (groupIndex: number, itemIndex: number): boolean => {
|
|
return (
|
|
selectedGroupIndex.value === groupIndex &&
|
|
selectedItemIndex.value === itemIndex
|
|
);
|
|
};
|
|
|
|
const searchHistory = async (): Promise<void> => {
|
|
const results = await $history.searchHistory(searchQuery.value);
|
|
history.value = results.map((item) =>
|
|
Object.assign(
|
|
new HistoryItem(
|
|
item.source,
|
|
item.content_type,
|
|
item.content,
|
|
item.favicon
|
|
),
|
|
{ id: item.id, timestamp: new Date(item.timestamp) }
|
|
)
|
|
);
|
|
};
|
|
|
|
const selectNext = (): void => {
|
|
const currentGroup = groupedHistory.value[selectedGroupIndex.value];
|
|
if (selectedItemIndex.value < currentGroup.items.length - 1) {
|
|
selectedItemIndex.value++;
|
|
} else if (selectedGroupIndex.value < groupedHistory.value.length - 1) {
|
|
selectedGroupIndex.value++;
|
|
selectedItemIndex.value = 0;
|
|
}
|
|
scrollToSelectedItem();
|
|
};
|
|
|
|
const selectPrevious = (): void => {
|
|
if (selectedItemIndex.value > 0) {
|
|
selectedItemIndex.value--;
|
|
} else if (selectedGroupIndex.value > 0) {
|
|
selectedGroupIndex.value--;
|
|
selectedItemIndex.value =
|
|
groupedHistory.value[selectedGroupIndex.value].items.length - 1;
|
|
}
|
|
scrollToSelectedItem();
|
|
};
|
|
|
|
const selectItem = (groupIndex: number, itemIndex: number): void => {
|
|
selectedGroupIndex.value = groupIndex;
|
|
selectedItemIndex.value = itemIndex;
|
|
scrollToSelectedItem();
|
|
};
|
|
|
|
const pasteSelectedItem = async (): Promise<void> => {
|
|
if (!selectedItem.value) return;
|
|
|
|
let content = selectedItem.value.content;
|
|
let contentType: string = selectedItem.value.content_type;
|
|
if (contentType === "image") {
|
|
try {
|
|
content = await $history.readImage({ filename: content });
|
|
} catch (error) {
|
|
console.error("Error reading image file:", error);
|
|
return;
|
|
}
|
|
}
|
|
await hideApp();
|
|
await $history.writeAndPaste({ content, contentType });
|
|
};
|
|
|
|
const truncateContent = (content: string): string => {
|
|
const maxWidth = 284;
|
|
const charWidth = 9;
|
|
const maxChars = Math.floor(maxWidth / charWidth);
|
|
return content.length > maxChars
|
|
? content.slice(0, maxChars - 3) + "..."
|
|
: content;
|
|
};
|
|
|
|
const hasFavicon = (str: string): boolean => {
|
|
return str.trim() !== "";
|
|
};
|
|
|
|
const isYoutubeWatchUrl = (url: string): boolean => {
|
|
return (
|
|
/^(https?:\/\/)?(www\.)?(youtube\.com|youtu\.be)\/watch\?v=[\w-]+/.test(
|
|
url
|
|
) || /^(https?:\/\/)?(www\.)?youtu\.be\/[\w-]+/.test(url)
|
|
);
|
|
};
|
|
|
|
const getYoutubeThumbnail = (url: string): string => {
|
|
let videoId;
|
|
if (url.includes("youtu.be")) {
|
|
videoId = url.split("youtu.be/")[1];
|
|
} else {
|
|
videoId = url.match(/[?&]v=([^&]+)/)?.[1];
|
|
}
|
|
return videoId
|
|
? `https://img.youtube.com/vi/${videoId}/maxresdefault.jpg`
|
|
: "https://via.placeholder.com/1280x720";
|
|
};
|
|
|
|
const getFaviconFromDb = (favicon: string): string => {
|
|
return `data:image/png;base64,${favicon}`;
|
|
};
|
|
|
|
const updateHistory = async (resetScroll: boolean = false): Promise<void> => {
|
|
const results = await $history.loadHistoryChunk(0, CHUNK_SIZE);
|
|
if (results.length > 0) {
|
|
const existingIds = new Set(history.value.map((item) => item.id));
|
|
const uniqueNewItems = results.filter((item) => !existingIds.has(item.id));
|
|
|
|
const processedNewItems = await Promise.all(
|
|
uniqueNewItems.map(async (item) => {
|
|
const historyItem = new HistoryItem(
|
|
item.source,
|
|
item.content_type,
|
|
item.content,
|
|
item.favicon
|
|
);
|
|
Object.assign(historyItem, {
|
|
id: item.id,
|
|
timestamp: new Date(item.timestamp),
|
|
});
|
|
|
|
if (historyItem.content_type === ContentType.Image) {
|
|
try {
|
|
const base64 = await $history.readImage({
|
|
filename: historyItem.content,
|
|
});
|
|
const size = Math.ceil((base64.length * 3) / 4);
|
|
imageSizes.value[historyItem.id] = formatFileSize(size);
|
|
|
|
const img = new Image();
|
|
img.src = `data:image/png;base64,${base64}`;
|
|
imageUrls.value[historyItem.id] = img.src;
|
|
|
|
await new Promise<void>((resolve) => {
|
|
img.onload = () => {
|
|
imageDimensions.value[
|
|
historyItem.id
|
|
] = `${img.width}x${img.height}`;
|
|
resolve();
|
|
};
|
|
img.onerror = () => {
|
|
imageDimensions.value[historyItem.id] = "Error";
|
|
resolve();
|
|
};
|
|
});
|
|
} catch (error) {
|
|
console.error("Error processing image:", error);
|
|
imageDimensions.value[historyItem.id] = "Error";
|
|
imageSizes.value[historyItem.id] = "Error";
|
|
}
|
|
}
|
|
return historyItem;
|
|
})
|
|
);
|
|
|
|
history.value = [...processedNewItems, ...history.value];
|
|
|
|
if (
|
|
resetScroll &&
|
|
resultsContainer.value?.osInstance()?.elements().viewport
|
|
) {
|
|
resultsContainer.value.osInstance()?.elements().viewport?.scrollTo({
|
|
top: 0,
|
|
behavior: "smooth",
|
|
});
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleSelection = (
|
|
groupIndex: number,
|
|
itemIndex: number,
|
|
shouldScroll: boolean = true
|
|
): void => {
|
|
selectedGroupIndex.value = groupIndex;
|
|
selectedItemIndex.value = itemIndex;
|
|
if (shouldScroll) scrollToSelectedItem();
|
|
};
|
|
|
|
const setupEventListeners = async (): Promise<void> => {
|
|
await listen("clipboard-content-updated", async () => {
|
|
lastUpdateTime.value = Date.now();
|
|
await updateHistory(true);
|
|
if (groupedHistory.value[0]?.items.length > 0) {
|
|
handleSelection(0, 0, false);
|
|
}
|
|
});
|
|
|
|
await listen("tauri://focus", async () => {
|
|
const currentTime = Date.now();
|
|
if (currentTime - lastUpdateTime.value > 0) {
|
|
const previousState = {
|
|
groupIndex: selectedGroupIndex.value,
|
|
itemIndex: selectedItemIndex.value,
|
|
scroll:
|
|
resultsContainer.value?.osInstance()?.elements().viewport
|
|
?.scrollTop || 0,
|
|
};
|
|
|
|
await updateHistory();
|
|
lastUpdateTime.value = currentTime;
|
|
handleSelection(previousState.groupIndex, previousState.itemIndex, false);
|
|
|
|
if (resultsContainer.value?.osInstance()?.elements().viewport?.scrollTo) {
|
|
resultsContainer.value.osInstance()?.elements().viewport?.scrollTo({
|
|
top: previousState.scroll,
|
|
behavior: "instant",
|
|
});
|
|
}
|
|
}
|
|
focusSearchInput();
|
|
});
|
|
|
|
await listen("tauri://blur", () => {
|
|
searchInput.value?.blur();
|
|
});
|
|
|
|
keyboard.prevent.down([Key.DownArrow], (event) => {
|
|
selectNext();
|
|
});
|
|
|
|
keyboard.prevent.down([Key.UpArrow], (event) => {
|
|
selectPrevious();
|
|
});
|
|
|
|
keyboard.prevent.down([Key.Enter], (event) => {
|
|
pasteSelectedItem();
|
|
});
|
|
|
|
keyboard.prevent.down([Key.Escape], (event) => {
|
|
hideApp();
|
|
});
|
|
|
|
switch (os.value) {
|
|
case "macos":
|
|
keyboard.prevent.down([Key.LeftMeta, Key.K], (event) => {});
|
|
|
|
keyboard.prevent.down([Key.RightMeta, Key.K], (event) => {});
|
|
break;
|
|
|
|
case "linux" || "windows":
|
|
keyboard.prevent.down([Key.LeftControl, Key.K], (event) => {});
|
|
|
|
keyboard.prevent.down([Key.RightControl, Key.K], (event) => {});
|
|
break;
|
|
}
|
|
};
|
|
|
|
const hideApp = async (): Promise<void> => {
|
|
await app.hide();
|
|
await window.getCurrentWindow().hide();
|
|
};
|
|
|
|
const focusSearchInput = (): void => {
|
|
nextTick(() => {
|
|
searchInput.value?.focus();
|
|
});
|
|
};
|
|
|
|
const onImageError = (): void => {
|
|
imageLoadError.value = true;
|
|
imageLoading.value = false;
|
|
};
|
|
|
|
watch([selectedGroupIndex, selectedItemIndex], () => {
|
|
scrollToSelectedItem();
|
|
});
|
|
|
|
watch(searchQuery, () => {
|
|
searchHistory();
|
|
});
|
|
|
|
onMounted(async () => {
|
|
try {
|
|
os.value = platform();
|
|
await loadHistoryChunk();
|
|
|
|
resultsContainer.value
|
|
?.osInstance()
|
|
?.elements()
|
|
?.viewport?.addEventListener("scroll", handleScroll);
|
|
|
|
await setupEventListeners();
|
|
} catch (error) {
|
|
console.error("Error during onMounted:", error);
|
|
}
|
|
});
|
|
|
|
watch([selectedGroupIndex, selectedItemIndex], () =>
|
|
scrollToSelectedItem(false)
|
|
);
|
|
|
|
const getFormattedDate = computed(() => {
|
|
if (!selectedItem.value?.timestamp) return "";
|
|
return new Intl.DateTimeFormat("en-US", {
|
|
dateStyle: "medium",
|
|
timeStyle: "medium",
|
|
}).format(selectedItem.value.timestamp);
|
|
});
|
|
|
|
const formatFileSize = (bytes: number): string => {
|
|
if (bytes === 0) return "0 Bytes";
|
|
const k = 1024;
|
|
const sizes = ["Bytes", "KB", "MB", "GB"];
|
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`;
|
|
};
|
|
|
|
const fetchPageMeta = async (url: string) => {
|
|
try {
|
|
const [title, ogImage] = (await invoke("fetch_page_meta", { url })) as [
|
|
string,
|
|
string | null
|
|
];
|
|
pageTitle.value = title;
|
|
if (ogImage) {
|
|
pageOgImage.value = ogImage;
|
|
}
|
|
} catch (error) {
|
|
console.error("Error fetching page meta:", error);
|
|
pageTitle.value = "Error loading title";
|
|
}
|
|
};
|
|
|
|
watch(
|
|
() => selectedItem.value,
|
|
(newItem) => {
|
|
if (newItem?.content_type === ContentType.Link) {
|
|
pageTitle.value = "Loading...";
|
|
pageOgImage.value = "";
|
|
fetchPageMeta(newItem.content);
|
|
} else {
|
|
pageTitle.value = "";
|
|
pageOgImage.value = "";
|
|
}
|
|
}
|
|
);
|
|
|
|
const getInfo = computed(() => {
|
|
if (!selectedItem.value) return null;
|
|
|
|
const baseInfo = {
|
|
source: selectedItem.value.source,
|
|
copied: selectedItem.value.timestamp,
|
|
};
|
|
|
|
const infoMap: Record<
|
|
ContentType,
|
|
() => InfoText | InfoImage | InfoFile | InfoLink | InfoColor | InfoCode
|
|
> = {
|
|
[ContentType.Text]: () => ({
|
|
...baseInfo,
|
|
content_type: ContentType.Text,
|
|
characters: selectedItem.value!.content.length,
|
|
words: selectedItem.value!.content.trim().split(/\s+/).length,
|
|
}),
|
|
[ContentType.Image]: () => ({
|
|
...baseInfo,
|
|
content_type: ContentType.Image,
|
|
dimensions: imageDimensions.value[selectedItem.value!.id] || "Loading...",
|
|
size: parseInt(imageSizes.value[selectedItem.value!.id] || "0"),
|
|
}),
|
|
[ContentType.File]: () => ({
|
|
...baseInfo,
|
|
content_type: ContentType.File,
|
|
path: selectedItem.value!.content,
|
|
filesize: 0,
|
|
}),
|
|
[ContentType.Link]: () => ({
|
|
...baseInfo,
|
|
content_type: ContentType.Link,
|
|
title: pageTitle.value,
|
|
url: selectedItem.value!.content,
|
|
characters: selectedItem.value!.content.length,
|
|
}),
|
|
[ContentType.Color]: () => {
|
|
const hex = selectedItem.value!.content;
|
|
const r = parseInt(hex.slice(1, 3), 16);
|
|
const g = parseInt(hex.slice(3, 5), 16);
|
|
const b = parseInt(hex.slice(5, 7), 16);
|
|
|
|
const rNorm = r / 255;
|
|
const gNorm = g / 255;
|
|
const bNorm = b / 255;
|
|
|
|
const max = Math.max(rNorm, gNorm, bNorm);
|
|
const min = Math.min(rNorm, gNorm, bNorm);
|
|
let h = 0,
|
|
s = 0;
|
|
const l = (max + min) / 2;
|
|
|
|
if (max !== min) {
|
|
const d = max - min;
|
|
s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
|
|
|
|
switch (max) {
|
|
case rNorm:
|
|
h = (gNorm - bNorm) / d + (gNorm < bNorm ? 6 : 0);
|
|
break;
|
|
case gNorm:
|
|
h = (bNorm - rNorm) / d + 2;
|
|
break;
|
|
case bNorm:
|
|
h = (rNorm - gNorm) / d + 4;
|
|
break;
|
|
}
|
|
h /= 6;
|
|
}
|
|
|
|
return {
|
|
...baseInfo,
|
|
content_type: ContentType.Color,
|
|
hex: hex,
|
|
rgb: `rgb(${r}, ${g}, ${b})`,
|
|
hsl: `hsl(${Math.round(h * 360)}, ${Math.round(s * 100)}%, ${Math.round(
|
|
l * 100
|
|
)}%)`,
|
|
};
|
|
},
|
|
[ContentType.Code]: () => ({
|
|
...baseInfo,
|
|
content_type: ContentType.Code,
|
|
language: selectedItem.value!.language ?? "Unknown",
|
|
lines: selectedItem.value!.content.split("\n").length,
|
|
}),
|
|
};
|
|
|
|
return infoMap[selectedItem.value.content_type]();
|
|
});
|
|
|
|
const infoRows = computed(() => {
|
|
if (!getInfo.value) return [];
|
|
|
|
const commonRows = [
|
|
{ label: "Source", value: getInfo.value.source, isUrl: false },
|
|
{
|
|
label: "Content Type",
|
|
value:
|
|
getInfo.value.content_type.charAt(0).toUpperCase() +
|
|
getInfo.value.content_type.slice(1),
|
|
isUrl: false,
|
|
},
|
|
];
|
|
|
|
const typeSpecificRows: Record<
|
|
ContentType,
|
|
Array<{ label: string; value: string | number; isUrl?: boolean }>
|
|
> = {
|
|
[ContentType.Text]: [
|
|
{ label: "Characters", value: (getInfo.value as InfoText).characters },
|
|
{ label: "Words", value: (getInfo.value as InfoText).words },
|
|
],
|
|
[ContentType.Image]: [
|
|
{ label: "Dimensions", value: (getInfo.value as InfoImage).dimensions },
|
|
{
|
|
label: "Image size",
|
|
value: formatFileSize((getInfo.value as InfoImage).size),
|
|
},
|
|
],
|
|
[ContentType.File]: [
|
|
{ label: "Path", value: (getInfo.value as InfoFile).path },
|
|
],
|
|
[ContentType.Link]: [
|
|
...((getInfo.value as InfoLink).title &&
|
|
(getInfo.value as InfoLink).title !== "Loading..."
|
|
? [{ label: "Title", value: (getInfo.value as InfoLink).title || "" }]
|
|
: []),
|
|
{ label: "URL", value: (getInfo.value as InfoLink).url, isUrl: true },
|
|
{ label: "Characters", value: (getInfo.value as InfoLink).characters },
|
|
],
|
|
[ContentType.Color]: [
|
|
{ label: "Hex", value: (getInfo.value as InfoColor).hex },
|
|
{ label: "RGB", value: (getInfo.value as InfoColor).rgb },
|
|
{ label: "HSL", value: (getInfo.value as InfoColor).hsl },
|
|
],
|
|
[ContentType.Code]: [
|
|
{ label: "Language", value: (getInfo.value as InfoCode).language },
|
|
{ label: "Lines", value: (getInfo.value as InfoCode).lines },
|
|
],
|
|
};
|
|
|
|
const specificRows = typeSpecificRows[getInfo.value.content_type].filter(
|
|
(row) => row.value !== ""
|
|
);
|
|
|
|
return [
|
|
...commonRows,
|
|
...specificRows,
|
|
{ label: "Copied", value: getFormattedDate.value },
|
|
];
|
|
});
|
|
</script>
|
|
|
|
<style scoped lang="scss">
|
|
@use "~/assets/css/index.scss";
|
|
</style>
|