Ziya/app/components/CexAnalysisCard.vue
rizary 67fb3a203e
feat: implement CEX analysis cards and real-time token monitoring
- Add TokenCard and CexAnalysisCard components for displaying token data
- Implement real-time Redis event streaming for token updates
- Add environment-based configuration system for dev/prod Redis servers
- Create comprehensive hunting ground dashboard with card management
- Add individual and bulk card removal functionality
- Implement browser integration for token details viewing
- Add timestamp utilities and proper type handling for Redis events
- Create production-ready configuration with 154.38.185.112 Redis server
- Add comprehensive documentation in README.md and CONTRIBUTORS.md
- Restructure project architecture with proper Electron-Vue integration

BREAKING CHANGE: Redis configuration now uses environment-based settings
2025-06-23 09:03:39 +07:00

538 lines
No EOL
18 KiB
Vue

<template>
<div
class="cex-analysis-card relative bg-base-100 hover:bg-base-200/50 transition-all duration-200 cursor-pointer border-b border-base-300 last:border-b-0"
:class="cardClass"
@click="$emit('click', token)"
>
<!-- Quick actions (visible on hover) -->
<div class="absolute top-2 right-2 z-10 opacity-0 group-hover:opacity-100 transition-opacity duration-200 flex gap-1">
<button
class="w-6 h-6 bg-base-300/90 hover:bg-error rounded text-base-content/60 hover:text-error-content transition-colors flex items-center justify-center text-xs"
title="Close token"
@click.stop="$emit('close', token)"
>
<Icon name="heroicons:x-mark" class="w-3 h-3" />
</button>
<button
class="w-6 h-6 bg-base-300/90 hover:bg-base-300 rounded text-base-content/60 hover:text-primary transition-colors flex items-center justify-center text-xs"
title="Hide token"
@click.stop="$emit('hide', token)"
>
<Icon name="heroicons:eye-slash" class="w-3 h-3" />
</button>
<button
class="w-6 h-6 bg-base-300/90 hover:bg-base-300 rounded text-base-content/60 hover:text-warning transition-colors flex items-center justify-center text-xs"
title="Watch token"
@click.stop="$emit('watch', token)"
>
<Icon name="heroicons:bookmark" class="w-3 h-3" />
</button>
</div>
<!-- Quick buy button (bottom right) -->
<div class="absolute bottom-2 right-2 z-10 opacity-0 group-hover:opacity-100 transition-opacity duration-200">
<button
class="bg-primary hover:bg-primary/80 text-primary-content px-2 py-1 rounded text-xs font-medium flex items-center gap-1 shadow-sm"
@click.stop="$emit('quick-buy', token)"
>
<Icon name="heroicons:bolt" class="w-3 h-3" />
Quick Buy
</button>
</div>
<!-- Main content -->
<div class="p-3 group">
<div class="flex items-start gap-3">
<!-- Token image/avatar -->
<div class="flex-shrink-0">
<div v-if="metadata?.image && !imageError" class="w-10 h-10 rounded-lg overflow-hidden bg-base-300 relative">
<img
:src="metadata.image"
:alt="token.name"
class="w-full h-full object-cover"
@error="handleImageError"
>
</div>
<div v-else-if="_metadataLoading" class="w-10 h-10 bg-base-300 rounded-lg flex items-center justify-center">
<div class="loading loading-spinner loading-xs text-primary" />
</div>
<div v-else-if="_metadataError" class="w-10 h-10 bg-error/20 rounded-lg flex items-center justify-center" :title="_metadataError">
<Icon name="heroicons:exclamation-triangle" class="w-4 h-4 text-error" />
</div>
<div v-else class="w-10 h-10 bg-gradient-to-br from-primary to-secondary rounded-lg flex items-center justify-center">
<span class="text-primary-content font-bold text-sm">{{ token.name?.charAt(0) || '?' }}</span>
</div>
</div>
<!-- Token info -->
<div class="flex-1 min-w-0">
<!-- Header with name and CEX badge -->
<div class="flex items-center gap-2 mb-1">
<h3 class="font-semibold text-sm text-base-content truncate">{{ token.name }}</h3>
<div class="flex items-center gap-1">
<!-- CEX badge with icon -->
<div
class="badge badge-xs flex items-center gap-1 px-2 py-1"
:class="cexBadgeClass"
>
<Icon :name="cexIcon" class="w-3 h-3" />
{{ cexDisplayName }}
</div>
<!-- CEX wallet type (if not main exchange name) -->
<span v-if="cexWalletType" class="text-xs text-base-content/50 font-mono">
{{ cexWalletType }}
</span>
</div>
</div>
<!-- Address -->
<div class="flex items-center gap-2 mb-2">
<span class="text-xs text-base-content/60 font-mono">{{ truncateAddress(mintAddress) }}</span>
<button
class="text-base-content/40 hover:text-primary transition-colors"
title="Copy address"
@click.stop="copyToClipboard(mintAddress)"
>
<Icon name="heroicons:clipboard-document" class="w-3 h-3" />
</button>
</div>
<!-- Analysis info and creator -->
<div class="flex items-center justify-between mb-2">
<div class="flex items-center gap-2">
<!-- Analysis type badge -->
<span class="badge badge-info badge-xs">
CEX ANALYSIS
</span>
<!-- Dev info (if not unknown_dev) -->
<div v-if="showDevInfo" class="flex items-center gap-1">
<Icon name="heroicons:user-circle" class="w-3 h-3 text-warning" />
<span class="text-xs text-warning font-medium">{{ token.dev_name }}</span>
</div>
<!-- Creator with graph tooltip -->
<div
class="relative"
@mouseenter="showGraphTooltip = true"
@mouseleave="showGraphTooltip = false"
>
<a
:href="`https://solscan.io/account/${token.creator}`"
target="_blank"
class="flex items-center gap-1 text-xs text-base-content/50 hover:text-primary transition-colors"
title="View creator on Solscan"
@click.stop
>
<Icon name="heroicons:user" class="w-3 h-3" />
{{ truncateAddress(token.creator) }}
</a>
<!-- Graph tooltip -->
<div
v-if="showGraphTooltip && graphNodes.length > 0"
class="absolute bottom-full left-0 mb-2 z-50 bg-base-200 rounded-lg shadow-lg border border-base-300 p-3 min-w-[200px]"
>
<div class="text-xs font-medium text-base-content mb-2">Connection Graph</div>
<div class="space-y-1">
<div v-for="node in graphNodes.slice(0, 5)" :key="node.id" class="flex items-center gap-2 text-xs">
<div class="w-2 h-2 rounded-full bg-primary" />
<span class="font-mono text-base-content/70">{{ truncateAddress(node.id) }}</span>
</div>
<div v-if="graphNodes.length > 5" class="text-xs text-base-content/50 text-center pt-1">
+{{ graphNodes.length - 5 }} more nodes
</div>
</div>
<div class="mt-2 pt-2 border-t border-base-300 text-xs text-base-content/60">
{{ token.node_count }} nodes, {{ token.edge_count }} edges
</div>
</div>
</div>
</div>
<!-- Analysis duration -->
<span class="text-xs text-base-content/50">
{{ analysisDuration }}
</span>
</div>
<!-- Social links (if available) -->
<div v-if="metadata && hasSocialLinks" class="flex items-center gap-1 mt-2">
<a
v-if="metadata.twitter"
:href="metadata.twitter"
target="_blank"
class="w-5 h-5 text-base-content/40 hover:text-info transition-colors"
title="Twitter"
@click.stop
>
<Icon name="simple-icons:x" class="w-3 h-3" />
</a>
<a
v-if="metadata.telegram"
:href="metadata.telegram"
target="_blank"
class="w-5 h-5 text-base-content/40 hover:text-info transition-colors"
title="Telegram"
@click.stop
>
<Icon name="simple-icons:telegram" class="w-3 h-3" />
</a>
<a
v-if="metadata.website"
:href="metadata.website"
target="_blank"
class="w-5 h-5 text-base-content/40 hover:text-info transition-colors"
title="Website"
@click.stop
>
<Icon name="heroicons:globe-alt" class="w-3 h-3" />
</a>
<a
v-if="metadata.discord"
:href="metadata.discord"
target="_blank"
class="w-5 h-5 text-base-content/40 hover:text-info transition-colors"
title="Discord"
@click.stop
>
<Icon name="simple-icons:discord" class="w-3 h-3" />
</a>
</div>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { computed, onMounted, ref } from 'vue';
import { truncateAddress as truncateAddr } from '~/utils/address';
import type { MaxDepthReachedData, TokenCexUpdatedData, TokenMetadata } from '../../types/redis-events';
import { fetchTokenMetadata } from '../utils/ipfs';
// Props
interface Props {
token: TokenCexUpdatedData | MaxDepthReachedData;
}
const props = defineProps<Props>();
// Emits
interface Emits {
(e: 'click' | 'hide' | 'watch' | 'quick-buy' | 'close', token: TokenCexUpdatedData | MaxDepthReachedData): void;
}
defineEmits<Emits>();
// Reactive data
const imageError = ref(false);
const mintAddress = ref<string>('');
const showGraphTooltip = ref(false);
// Simple metadata state management
const metadata = ref<TokenMetadata | null>(null);
const _metadataLoading = ref(false);
const _metadataError = ref<string | null>(null);
// CEX mapping utilities
const getCexInfo = (cexName: string) => {
const name = cexName.toLowerCase();
// Extract base exchange name and wallet type
if (name.includes('coinbase')) {
const type = name.replace('coinbase_', '').replace('coinbase', '');
return {
baseName: 'Coinbase',
walletType: type ? type.toUpperCase() : '',
color: 'badge-info',
icon: 'simple-icons:coinbase'
};
}
if (name.includes('binance')) {
const type = name.replace('binance_', '').replace('binance', '');
return {
baseName: 'Binance',
walletType: type ? type.toUpperCase() : '',
color: 'badge-warning',
icon: 'simple-icons:binance'
};
}
if (name.includes('okx')) {
const type = name.replace('okx_', '').replace('okx', '');
return {
baseName: 'OKX',
walletType: type ? type.toUpperCase() : '',
color: 'badge-primary',
icon: 'heroicons:building-office'
};
}
if (name.includes('kraken')) {
const type = name.replace('kraken_', '').replace('kraken', '');
return {
baseName: 'Kraken',
walletType: type ? type.toUpperCase() : '',
color: 'badge-secondary',
icon: 'heroicons:building-office-2'
};
}
if (name.includes('mexc')) {
const type = name.replace('mexc_', '').replace('mexc', '');
return {
baseName: 'MEXC',
walletType: type ? type.toUpperCase() : '',
color: 'badge-accent',
icon: 'heroicons:building-office'
};
}
if (name.includes('bitget')) {
const type = name.replace('bitget_', '').replace('bitget', '');
return {
baseName: 'Bitget',
walletType: type ? type.toUpperCase() : '',
color: 'badge-info',
icon: 'heroicons:building-office'
};
}
if (name.includes('gateio') || name.includes('gate.io')) {
const type = name.replace('gateio_', '').replace('gateio', '');
return {
baseName: 'Gate.io',
walletType: type ? type.toUpperCase() : '',
color: 'badge-primary',
icon: 'heroicons:building-office'
};
}
if (name.includes('bybit')) {
const type = name.replace('bybit_', '').replace('bybit', '');
return {
baseName: 'Bybit',
walletType: type ? type.toUpperCase() : '',
color: 'badge-warning',
icon: 'heroicons:building-office'
};
}
if (name.includes('bitfinex')) {
const type = name.replace('bitfinex_', '').replace('bitfinex', '');
return {
baseName: 'Bitfinex',
walletType: type ? type.toUpperCase() : '',
color: 'badge-success',
icon: 'heroicons:building-office'
};
}
if (name.includes('kucoin')) {
const type = name.replace('kucoin_', '').replace('kucoin', '');
return {
baseName: 'KuCoin',
walletType: type ? type.toUpperCase() : '',
color: 'badge-accent',
icon: 'heroicons:building-office'
};
}
if (name.includes('poloniex')) {
const type = name.replace('poloniex_', '').replace('poloniex', '');
return {
baseName: 'Poloniex',
walletType: type ? type.toUpperCase() : '',
color: 'badge-neutral',
icon: 'heroicons:building-office'
};
}
if (name.includes('lbank')) {
return {
baseName: 'LBank',
walletType: '',
color: 'badge-neutral',
icon: 'heroicons:building-office'
};
}
if (name.includes('debridge')) {
return {
baseName: 'DeBridge',
walletType: 'VAULT',
color: 'badge-secondary',
icon: 'heroicons:shield-check'
};
}
if (name.includes('revolut')) {
return {
baseName: 'Revolut',
walletType: 'HOT',
color: 'badge-info',
icon: 'heroicons:credit-card'
};
}
if (name.includes('bitstamp')) {
return {
baseName: 'BitStamp',
walletType: 'HOT',
color: 'badge-success',
icon: 'heroicons:building-office'
};
}
if (name.includes('stakecom')) {
return {
baseName: 'Stake.com',
walletType: 'HOT',
color: 'badge-warning',
icon: 'heroicons:fire'
};
}
// Default for unknown exchanges
return {
baseName: cexName.replace(/_/g, ' ').toUpperCase(),
walletType: '',
color: 'badge-neutral',
icon: 'heroicons:building-office'
};
};
// Computed properties
const cexInfo = computed(() => getCexInfo(props.token.cex_name));
const cardClass = computed(() => {
const baseClass = 'h-[140px] min-h-[140px]';
// Use CEX-specific border color
if (cexInfo.value.color.includes('info')) {
return `${baseClass} border-l-2 border-l-info`;
} else if (cexInfo.value.color.includes('warning')) {
return `${baseClass} border-l-2 border-l-warning`;
} else if (cexInfo.value.color.includes('success')) {
return `${baseClass} border-l-2 border-l-success`;
} else if (cexInfo.value.color.includes('primary')) {
return `${baseClass} border-l-2 border-l-primary`;
} else if (cexInfo.value.color.includes('secondary')) {
return `${baseClass} border-l-2 border-l-secondary`;
} else if (cexInfo.value.color.includes('accent')) {
return `${baseClass} border-l-2 border-l-accent`;
}
return `${baseClass} border-l-2 border-l-neutral`;
});
const cexBadgeClass = computed(() => cexInfo.value.color);
const cexDisplayName = computed(() => cexInfo.value.baseName);
const cexWalletType = computed(() => cexInfo.value.walletType);
const cexIcon = computed(() => cexInfo.value.icon);
const showDevInfo = computed(() => {
return props.token.dev_name && props.token.dev_name !== 'unknown_dev';
});
const analysisDuration = computed(() => {
const createdAt = typeof props.token.created_at === 'string'
? parseInt(props.token.created_at)
: props.token.created_at;
const updatedAt = typeof props.token.updated_at === 'string'
? parseInt(props.token.updated_at)
: props.token.updated_at;
const durationSeconds = updatedAt - createdAt;
if (durationSeconds < 60) {
return `${durationSeconds}s analysis`;
} else if (durationSeconds < 3600) {
const minutes = Math.floor(durationSeconds / 60);
return `${minutes}m analysis`;
} else {
const hours = Math.floor(durationSeconds / 3600);
const minutes = Math.floor((durationSeconds % 3600) / 60);
return `${hours}h ${minutes}m analysis`;
}
});
const graphNodes = computed(() => {
try {
if (props.token.graph && typeof props.token.graph === 'object') {
const graph = props.token.graph as { graph?: { nodes?: Array<{ id: string }> } };
if (graph.graph && graph.graph.nodes) {
return graph.graph.nodes;
}
}
return [];
} catch (error) {
console.error('Error parsing graph data:', error);
return [];
}
});
const hasSocialLinks = computed(() => {
return metadata.value && (
metadata.value.twitter ||
metadata.value.telegram ||
metadata.value.website ||
metadata.value.discord
);
});
// Methods
const truncateAddress = (address: string): string => {
return truncateAddr(address);
};
const copyToClipboard = async (text: string): Promise<void> => {
try {
await navigator.clipboard.writeText(text);
// You could add a toast notification here
} catch (err) {
console.error('Failed to copy to clipboard:', err);
}
};
const handleImageError = (): void => {
imageError.value = true;
};
// Load metadata on mount
onMounted(async () => {
// Set mint address (now it's already a string)
mintAddress.value = props.token.mint;
// Load metadata if URI exists
if (props.token.uri) {
_metadataLoading.value = true;
_metadataError.value = null;
try {
const result = await fetchTokenMetadata(props.token.uri);
metadata.value = result;
} catch (error) {
const errorMessage = error instanceof Error ? error.message : 'Failed to fetch metadata';
_metadataError.value = errorMessage;
} finally {
_metadataLoading.value = false;
}
}
});
</script>
<style scoped>
.cex-analysis-card {
transition: all 0.2s cubic-bezier(0.4, 0, 0.2, 1);
}
.cex-analysis-card:hover {
transform: translateY(-1px);
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
.group:hover .opacity-0 {
opacity: 1;
}
</style>