Transaction

90c3e9f41c8b8fb840d9bac8a875226b6ee2b7b00c838ebc0cb7f8a3c04e793c
Timestamp (utc)
2025-02-09 20:41:44
Fee Paid
0.00009922 BSV
(
0.03846818 BSV
-
0.03836896 BSV
)
Fee Rate
499.4 sat/KB
Version
1
Confirmations
53,184
Size Stats
19,867 B

2 Outputs

Total Output:
0.03836896 BSV
  • jMÃŒL{"name":"EncodedFiles","files":[{"name":"ClassCoreGlobalsF.html","size":14660,"type":"text/html","content":"class CoreGlobals {
  constructor() {
    // Define Twemoji implementation directly
    this.twemoji = {
      parse: (content) => {
        return content.replace(/[\u{1F300}-\u{1F9FF}]|[\u{2600}-\u{26FF}]/gu, (match) => {
          const codePoint = match.codePointAt(0).toString(16);
          return `<img class="emoji" draggable="false" alt="${match}" src="https://cdnjs.cloudflare.com/ajax/libs/twemoji/14.0.2/svg/${codePoint}.svg">`;
        });
      }
    };

    // Store emoji codes (using Unicode escapes) to prevent encoding issues
    this.EMOJI_CODES = {
      INBOX: '\u{1F4E5}',      // 📥
      SENT: '\u{1F4E4}',       // 📤
      JUNK: '\u{1F5D1}',       // 🗑
      FAVORITES: '\u{2B50}',    // ⭐
      ADD: '\u{2795}',         // ➕
      FOLDER: '\u{1F4C1}',     // 📁
      CONTACTS: '\u{1F465}'    // 👥
    };

    // Define default folders
    this.FOLDERS = {
      inbox: {
        name: 'Inbox',
        storageKey: 'messageInbox',
        icon: this.EMOJI_CODES.INBOX
      },
      sent: {
        name: 'Sent',
        storageKey: 'sentMessages',
        icon: this.EMOJI_CODES.SENT
      },
      junk: {
        name: 'Junk',
        storageKey: 'junkTransactions',
        icon: this.EMOJI_CODES.JUNK
      },
      favorites: {
        name: 'Favorites',
        storageKey: 'favoriteMessages',
        icon: this.EMOJI_CODES.FAVORITES
      }
    };

    // Global state
    this.globalState = {
      currentView: null,
      currentFolder: null,
      transactionListener: null
    };

    // Inject base CSS (for emoji and unified folder display)
    const style = document.createElement('style');
    style.textContent = `
      /* Emoji Styles */
      img.emoji {
        height: 1.2em;
        width: 1.2em;
        margin: 0 .05em 0 .1em;
        vertical-align: -0.1em;
      }
      .folder-icon img.emoji {
        margin-right: 0.5rem;
      }
      
      /* Unified Folder Display Styles */
      .folder-header {
        display: flex;
        justify-content: space-between;
        align-items: center;
        padding: 1rem;
        border-bottom: 1px solid #e5e7eb;
      }
      .folder-content {
        padding: 1rem;
      }
      .message-item {
        padding: 1rem;
        border-bottom: 1px solid #e5e7eb;
        display: flex;
        justify-content: space-between;
        align-items: center;
        cursor: pointer;
      }
      .message-item div {
        font-family: monospace;
      }
      .btn {
        padding: 0.5rem 1rem;
        border-radius: 0.375rem;
        border: none;
        cursor: pointer;
        font-weight: 500;
        font-size: 0.875rem;
      }
      .btn-primary {
        background: #3B82F6;
        color: white;
      }
      .btn-danger {
        background: #EF4444;
        color: white;
      }
      /* Navigation items for consistency */
      .nav-item {
        padding: 0.5rem;
        cursor: pointer;
      }
    `;
    document.head.appendChild(style);
  }

  // Helper: Render emojis using our twemoji parser
  renderEmoji(content) {
    return this.twemoji.parse(content);
  }

  // Set up global functions and inject globals into window
  injectGlobals() {
    window.FOLDERS = this.FOLDERS;
    window.EMOJI_CODES = this.EMOJI_CODES;
    window.globalState = this.globalState;

    // Get folders from localStorage or use defaults
    window.getFolders = () => {
      return JSON.parse(localStorage.getItem('folders') || JSON.stringify(this.FOLDERS));
    };

    window.getContactName = (address) => {
      const contacts = JSON.parse(localStorage.getItem('contacts') || '{}');
      return contacts[address] || address;
    };

    // Global function: Show Inbox (calls setupFolder)
    window.showInbox = () => {
      window.globalState.currentFolder = 'inbox';
      window.setupFolder('inbox');
    };

    // Global function: Show Sent
    window.showSent = () => {
      window.globalState.currentFolder = 'sent';
      window.setupFolder('sent');
    };

    // Global: Save a sent message
    window.saveSentMessage = (recipient, txid) => {
      const sentMessages = JSON.parse(localStorage.getItem('sentMessages') || '[]');
      sentMessages.push({
        recipient,
        txid,
        timestamp: Date.now()
      });
      localStorage.setItem('sentMessages', JSON.stringify(sentMessages));
      if (window.globalState.currentFolder === 'sent') {
        window.setupFolder('sent');
      }
    };

    // Global: Save contact
    window.saveContact = (address, name) => {
      const contacts = JSON.parse(localStorage.getItem('contacts') || '{}');
      contacts[address] = name;
      localStorage.setItem('contacts', JSON.stringify(contacts));
    };

    // Global: Delete message from a folder
    window.deleteMessage = (txid, folderKey) => {
      const folder = this.FOLDERS[folderKey || window.globalState.currentFolder];
      if (!folder) return;
      const stored = JSON.parse(localStorage.getItem(folder.storageKey) || '[]');
      const updated = stored.filter(tx => tx.txid !== txid);
      localStorage.setItem(folder.storageKey, JSON.stringify(updated));
      window.setupFolder(folderKey || window.globalState.currentFolder);
    };

    // Global: Refresh inbox
    window.refreshInbox = async () => {
      const address = localStorage.getItem('walletAddress');
      if (!address) {
        console.error('No wallet address found');
        return;
      }
      const loadingIndicator = document.createElement('div');
      loadingIndicator.textContent = 'Fetching message transactions...';
      loadingIndicator.style.padding = '1rem';
      document.getElementById('messageList').prepend(loadingIndicator);
      try {
        const transactions = await fetchAddressTransactions(address);
        const { messages, newCount } = await processTransactions(transactions, address);
        if (newCount > 0) {
          window.modalManager?.showAlert(`Found ${newCount} new message(s)`);
        }
        window.setupFolder('inbox');
      } catch (error) {
        console.error('Refresh failed:', error);
        window.modalManager?.showAlert('Failed to refresh: ' + error.message, 'error');
      } finally {
        loadingIndicator.remove();
      }
    };

    // Global: Setup navigation (folder links)
window.setupNavigation = () => {
  const nav = document.querySelector('nav');
  const folders = window.getFolders();
  
  // Build system navigation items
  const systemNav = `
    <a class="nav-item" onclick="showInbox()">
      <span class="folder-icon">${window.coreGlobals.renderEmoji(window.EMOJI_CODES.INBOX)}</span> Inbox
    </a>
    <a class="nav-item" onclick="showSent()">
      <span class="folder-icon">${window.coreGlobals.renderEmoji(window.EMOJI_CODES.SENT)}</span> Sent
    </a>
    <a class="nav-item" onclick="showContacts()">
      <span class="folder-icon">${window.coreGlobals.renderEmoji(window.EMOJI_CODES.CONTACTS)}</span> Contacts
    </a>
  `;
  
  // Custom folders
  const customFolders = Object.entries(folders)
    .filter(([key]) => !['inbox', 'sent'].includes(key))
    .map(([key, folder]) => `
      <a class="nav-item" onclick="showFolder('${key}')">
        <span class="folder-icon">${window.coreGlobals.renderEmoji(folder.icon)}</span> ${folder.name}
      </a>
    `).join('');
    
  // Add Folder button
  const addFolderButton = `
    <a class="nav-item add-folder" onclick="showAddFolderModal()">
      <span class="folder-icon">${window.coreGlobals.renderEmoji(window.EMOJI_CODES.ADD)}</span> Add Folder
    </a>
  `;
  
  nav.innerHTML = systemNav + customFolders + addFolderButton;
};

    // Unified folder display for Inbox, Sent, and others
window.setupFolder = (folderKey) => {
        const folders = window.getFolders();
        const folder = folders[folderKey];
        if (!folder) return;
        const messages = JSON.parse(localStorage.getItem(folder.storageKey) || '[]');
        const messageList = document.getElementById('messageList');
        if (!messageList) return;
        
        const headerHTML = `
            <div class="folder-header">
                <div class="folder-header-left" style="display: flex; align-items: center; gap: 8px;">
                    <span class="folder-icon">${window.coreGlobals.renderEmoji(folder.icon)}</span>
                    <h2 style="margin: 0;">${folder.name} (${messages.length})</h2>
                </div>
                ${folderKey === 'inbox' ? '<button onclick="refreshInbox()" class="btn btn-secondary">Refresh</button>' : ''}
            </div>
        `;

        const formatDate = (timestamp) => {
            if (!timestamp || isNaN(new Date(timestamp).getTime())) {
                return '';
            }
            return new Date(timestamp).toLocaleString();
        };

        const truncateText = (text, maxLength = 280) => {
            if (!text) return '';
            return text.length > maxLength ? text.slice(0, maxLength) + '...' : text;
        };

        const messageItemsHTML = messages.length === 0
            ? `<div style="text-align: center; padding: 2rem; color: #6b7280;">No messages in ${folder.name}</div>`
            : messages.sort((a, b) => b.timestamp - a.timestamp)
                .map(message => {
                    const date = formatDate(message.timestamp);
                    const sender = message.sender ? window.getContactName(message.sender) : 'Unknown Sender';
                    const preview = truncateText(message.messages?.[0] || '');
                    
                    return `
                        <div class="message-item" onclick="window.messageHandler.showMessage('${message.txid}')" 
                             style="display: flex; justify-content: space-between; align-items: start; padding: 0.75rem; border-bottom: 1px solid #e5e7eb;">
                            <div style="flex: 1; min-width: 0; margin-right: 1rem;">
                                <div style="font-weight: 500; margin-bottom: 0.25rem;">
                                    ${sender}
                                </div>
                                <div class="txid" style="font-family: monospace; font-size: 0.875rem; color: #6b7280; margin-bottom: 0.25rem;">
                                    ${message.txid}
                                </div>
                                <div style="color: #6b7280; font-size: 0.875rem; margin-bottom: 0.25rem;">
                                    ${date}
                                </div>
                                <div style="color: #374151; font-size: 0.875rem;">
                                    ${preview}
                                </div>
                            </div>
                            <div class="message-actions" style="display: flex; gap: 0.5rem; align-items: center; position: relative;">
                                <button 
                                    class="btn btn-secondary" 
                                    onclick="event.stopPropagation(); 
                                        const moveMenu = this.nextElementSibling;
                                        moveMenu.style.display = moveMenu.style.display === 'none' ? 'block' : 'none';"
                                    style="padding: 0.25rem 0.5rem; font-size: 0.875rem;"
                                >
                                    📁 Move
                                </button>
                                <div class="move-menu" style="display: none; position: absolute; right: 0; top: 100%; z-index: 50; background: white; border: 1px solid #e5e7eb; border-radius: 4px; box-shadow: 0 2px 4px rgba(0,0,0,0.1);">
                                    <div class="move-menu-item" onclick="event.stopPropagation(); window.messageHandler.moveMessage('${message.txid}', 'junk');" style="padding: 8px 16px; cursor: pointer;">
                                        ${window.coreGlobals.renderEmoji(folders.junk.icon)} Move to Junk
                                    </div>
                                    <div class="move-menu-item" onclick="event.stopPropagation(); window.messageHandler.moveMessage('${message.txid}', 'favorites');" style="padding: 8px 16px; cursor: pointer;">
                                        ${window.coreGlobals.renderEmoji(folders.favorites.icon)} Move to Favorites
                                    </div>
                                </div>
                                <button 
                                    onclick="event.stopPropagation(); deleteMessage('${message.txid}', '${folderKey}')" 
                                    class="btn btn-danger"
                                    style="padding: 0.25rem 0.5rem; font-size: 0.875rem;"
                                >
                                    Delete
                                </button>
                            </div>
                        </div>
                    `;
                }).join('');

        messageList.innerHTML = `
            ${headerHTML}
            <div class="message-content-container">
                ${messageItemsHTML}
            </div>
        `;

        // Add global click handler to close move menus when clicking outside
        document.addEventListener('click', (event) => {
            if (!event.target.closest('.move-button-container')) {
                document.querySelectorAll('.move-menu').forEach(menu => {
                    menu.style.display = 'none';
                });
            }
        });
    };

    // Global: Switch folder view
    window.showFolder = (folderKey) => {
      window.globalState.currentFolder = folderKey;
      window.setupFolder(folderKey);
    };

    // Global: Show contacts (if ContactsManager is defined)
    window.showContacts = () => {
      if (window.contactsManager && typeof window.contactsManager.showContacts === 'function') {
        window.contactsManager.showContacts();
      } else {
        console.warn('ContactsManager is not available.');
      }
    };
  }

  initialize() {
    this.injectGlobals();
    this.setupEventListeners();
    console.log('Core globals initialized successfully');
  }

  setupEventListeners() {
    // Listen for localStorage changes from other tabs
    window.addEventListener('storage', (e) => {
      if (e.key === 'messageInbox' || e.key === 'sentMessages') {
        if (window.globalState.currentFolder === e.key) {
          window.setupFolder(window.globalState.currentFolder);
        }
      }
    });
  }
}

// Instantiate and initialize CoreGlobals
window.coreGlobals = new CoreGlobals();
window.coreGlobals.initialize();
"}]}
    https://whatsonchain.com/tx/90c3e9f41c8b8fb840d9bac8a875226b6ee2b7b00c838ebc0cb7f8a3c04e793c