jM0X{"name":"EncodedFiles","files":[{"name":"messagehandlerAC2.html","size":16848,"type":"text/html","content":"class MessageHandler {
  constructor() {
    this.fetcher = new BlockchainFetcher();
    // We rely on the CoreGlobals styles for unified styling

    // Close any open dropdowns when clicking outside
    document.addEventListener('click', (event) => {
      const dropdowns = document.querySelectorAll('.dropdown-menu');
      const buttons = document.querySelectorAll('.dropdown-toggle');
      
      if (event.target.closest('.dropdown-menu') || Array.from(buttons).includes(event.target)) {
        return;
      }
      
      dropdowns.forEach(menu => {
        menu.style.display = 'none';
      });
    });
  }

  // --------------------------
  // Unified Message Flow
  // --------------------------
  async openMessage(txid) {
    try {
      // Fetch transaction data (same for inbox and sent)
      const rawData = await this.fetcher.fetchTransaction(txid);
      // Determine if the current folder is "sent"
      const isSentMessage = window.globalState.currentFolder === 'sent';
      // Use the same parsing & rendering flow for both inbox and sent messages
      await this.showContent(rawData, txid, isSentMessage);
    } catch (error) {
      console.error("Error displaying message:", error);
      alert("Error displaying message: " + error.message);
    }
  }

  // --------------------------
  // Sent Folder Specific Rendering
  // --------------------------
  updateSentFolder() {
    const messageList = document.getElementById("messageList");
    const sentMessages = JSON.parse(localStorage.getItem("sentMessages") || "[]");

console.log("Raw sent messages from storage:", sentMessages);
    
    // Use the unified folder-header and folder-content classes
    messageList.innerHTML = `
      <div class="folder-header">
        <div class="folder-header-left" style="display: flex; align-items: center; gap: 8px;">
          <span class="folder-icon">${window.coreGlobals.renderEmoji(window.EMOJI_CODES.SENT)}</span>
          <h2 style="margin: 0;">Sent (${sentMessages.length})</h2>
        </div>
      </div>
      <div class="folder-content">
        ${sentMessages.map(msg => `
          <div class="message-item" onclick="window.messageHandler.openMessage(${JSON.stringify(msg.txid)})">
            <div class="txid">${msg.txid}</div>
            <div class="message-time">
              ${new Date(msg.timestamp).toLocaleString()}
            </div>
            ${msg.messageData?.rewardData ? 
              `<div class="reward-badge">💰 Reward: ${msg.messageData.rewardData.actualReward} sats</div>` 
              : ''
            }
            <button 
              class="btn btn-danger" 
              onclick="window.messageHandler.deleteSentMessage(${JSON.stringify(msg.txid)}); event.stopPropagation();"
            >
              ❌
            </button>
          </div>
        `).join("")}
      </div>
    `;
  }

  deleteSentMessage(txid) {
    const sentMessages = JSON.parse(localStorage.getItem("sentMessages") || "[]");
    const updated = sentMessages.filter(msg => msg.txid !== txid);
    localStorage.setItem("sentMessages", JSON.stringify(updated));
    this.updateSentFolder();
  }

  // --------------------------
  // Inbox Message Flow
  // --------------------------
  async showMessage(txid) {
    // For backwards compatibility, call the unified method.
    await this.openMessage(txid);
  }

  // --------------------------
  // Shared Content Display (Used for both Inbox & Sent)
  // --------------------------
  async showContent(rawData, txid, isSentMessage = false) {
    let content;
    let messageData = null;
    try {
      // Attempt to extract JSON from the raw hex data
      const jsonStart = rawData.indexOf('7b22'); // hex for '{"'
      if (jsonStart === -1) {
        throw new Error('No JSON data found');
      }
      const jsonHex = rawData.slice(jsonStart);
      const bytes = new Uint8Array(
        jsonHex.match(/.{1,2}/g).map(byte => parseInt(byte, 16))
      );
      const jsonString = new TextDecoder().decode(bytes);
      messageData = JSON.parse(jsonString);
      if (!messageData.files || !messageData.files[0]) {
        throw new Error('Invalid message format');
      }
      content = {
        data: messageData.files[0].content,
        type: messageData.files[0].type || 'text/plain',
        name: messageData.files[0].name
      };
    } catch (error) {
      // Fallback to treating rawData as plain text if JSON parsing fails
      content = {
        data: rawData,
        type: 'text/plain',
        name: 'message.txt'
      };
    }

    const mailtoLink = this.createMailtoLink(txid, content.data, content.type);
    const messageList = document.getElementById('messageList');

    // Choose the appropriate reward functionality based on the folder
    const rewardFunction = isSentMessage ? 'checkAndShowRefundOption' : 'getAndShowRewardData';
    const rewardButtonText = isSentMessage ? 'Check/Refund Reward' : 'Check/Claim Reward';

    // Render the view using CoreGlobals' unified classes
    messageList.innerHTML = `
      <div class="folder-header">
        <div class="folder-header-left" style="display: flex; align-items: center;">
          <button class="btn btn-secondary" onclick="closeMessageList()">Back to ${isSentMessage ? 'Sent' : 'Inbox'}</button>
        </div>
        <div class="folder-header-right" style="display: flex; gap: 0.5rem; align-items: center;">
          <div class="dropdown">
            <button class="btn btn-primary" onclick="window.messageHandler.toggleShareDropdown(event)">Share via Email</button>
            <div class="dropdown-menu">
              <a class="dropdown-item" href="${mailtoLink.mailto}" target="_blank">Default Email Client</a>
              <a class="dropdown-item" href="${mailtoLink.gmail}" target="_blank">Gmail</a>
              <a class="dropdown-item" href="${mailtoLink.outlook}" target="_blank">Outlook</a>
              <a class="dropdown-item" href="${mailtoLink.yahoo}" target="_blank">Yahoo Mail</a>
              <button class="dropdown-item" onclick="window.messageHandler.copyToClipboard(${JSON.stringify(mailtoLink.content)})">Copy Share Link</button>
            </div>
          </div>
          <button class="btn btn-primary" onclick="window.messageHandler.${rewardFunction}(${JSON.stringify(txid)})">
            ${rewardButtonText}
          </button>
        </div>
      </div>
      <div class="folder-content">
        <div class="message-info" style="margin-bottom: 1rem;">
          <div><strong>File:</strong> ${content.name}</div>
          <div><strong>Type:</strong> ${content.type}</div>
        </div>
        <div class="message-content">
          ${this.formatContent(content)}
        </div>
        <div id="rewardContainer" style="margin-top: 1rem;"></div>
      </div>
    `;
  }

  // --------------------------
  // Reward Handling Methods
  // --------------------------
  async getAndShowRewardData(txid) {
    const rewardContainer = document.getElementById('rewardContainer');
    rewardContainer.innerHTML = '<div class="loading">Checking for reward...</div>';

    try {
      const resultDiv = document.createElement('div');
      resultDiv.id = 'rewardResult';
      
      await window.testGetRewardData(txid, resultDiv);
      
      const rewardDataText = resultDiv.querySelector('pre')?.textContent;
      if (!rewardDataText) {
        throw new Error('No reward data found');
      }

      const rewardData = JSON.parse(rewardDataText);
      
      rewardContainer.innerHTML = `
        <div class="reward-claim-container">
          <div class="reward-info">
            <h3>💰 Reward Found!</h3>
            <p>Amount: ${rewardData.actualReward} satoshis</p>
            <p>Address: ${rewardData.rewardAddress}</p>
            <button class="btn btn-primary" id="claimButton">Claim Reward</button>
            <div id="claimStatus"></div>
          </div>
        </div>
      `;

      document.getElementById('claimButton').onclick = () => this.claimReward(txid, rewardData);

    } catch (error) {
      console.error('Error checking reward:', error);
      rewardContainer.innerHTML = `
        <div class="reward-claim-container">
          <div class="error-message">
            Error checking reward: ${error.message}
            <button class="btn btn-primary" onclick="window.messageHandler.getAndShowRewardData(${JSON.stringify(txid)})">
              Try Again
            </button>
          </div>
        </div>
      `;
    }
  }

  async checkAndShowRefundOption(txid) {
    const rewardContainer = document.getElementById('rewardContainer');
    rewardContainer.innerHTML = '<div class="loading">Checking for refundable reward...</div>';

    try {
      const resultDiv = document.createElement('div');
      resultDiv.id = 'rewardResult';
      
      await window.testGetRewardData(txid, resultDiv);
      
      const rewardDataText = resultDiv.querySelector('pre')?.textContent;
      if (!rewardDataText) {
        throw new Error('No reward data found');
      }

      const rewardData = JSON.parse(rewardDataText);
      
      rewardContainer.innerHTML = `
        <div class="reward-claim-container">
          <div class="reward-info">
            <h3>💰 Refundable Reward Found!</h3>
            <p>Amount: ${rewardData.actualReward} satoshis</p>
            <p>Address: ${rewardData.rewardAddress}</p>
            <button class="btn btn-primary" id="refundButton">Refund Reward</button>
            <div id="refundStatus"></div>
          </div>
        </div>
      `;

      document.getElementById('refundButton').onclick = () => this.handleRefund(txid, rewardData);

    } catch (error) {
      console.error('Error checking refundable reward:', error);
      rewardContainer.innerHTML = `
        <div class="reward-claim-container">
          <div class="error-message">
            Error checking refundable reward: ${error.message}
            <button class="btn btn-primary" onclick="window.messageHandler.checkAndShowRefundOption(${JSON.stringify(txid)})">
              Try Again
            </button>
          </div>
        </div>
      `;
    }
  }

  async claimReward(txid, rewardData) {
    const statusDiv = document.getElementById('claimStatus');
    const claimButton = document.getElementById('claimButton');
    
    try {
      if (!txid || !rewardData?.amount || !rewardData?.rewardPrivateKeyWIF) {
        throw new Error("Missing required reward data");
      }

      const recipientAddress = localStorage.getItem("walletAddress");
      if (!recipientAddress) {
        throw new Error("Please restore your wallet first");
      }

      claimButton.disabled = true;
      claimButton.textContent = 'Claiming...';
      statusDiv.innerHTML = '<div class="loading">Processing claim...</div>';
      
      const result = await window.claimReward(
        txid,
        1,
        rewardData.amount,
        rewardData.rewardPrivateKeyWIF,
        recipientAddress
      );

      statusDiv.innerHTML = `
        <div class="success-message">
          ✅ Reward claimed successfully!<br>
          Transaction ID: ${result}
        </div>
      `;
      
      claimButton.style.display = 'none';
      await window.refreshInbox?.();

    } catch (error) {
      console.error('Claim error:', error);
      statusDiv.innerHTML = `
        <div class="error-message">
          ❌ Failed to claim reward: ${error.message}
        </div>
      `;
      claimButton.disabled = false;
      claimButton.textContent = 'Try Again';
    }
  }

  async handleRefund(txid, rewardData) {
    const statusDiv = document.getElementById('refundStatus');
    const refundButton = document.getElementById('refundButton');
    
    try {
      const senderPrivateKey = localStorage.getItem("privateKey");
      if (!senderPrivateKey) {
        throw new Error("Please restore your wallet first");
      }

      refundButton.disabled = true;
      refundButton.textContent = 'Refunding...';
      statusDiv.innerHTML = '<div class="loading">Processing refund...</div>';

      const result = await window.refundReward(
        txid,
        1,
        rewardData.amount,
        rewardData.rewardPrivateKeyWIF,
        localStorage.getItem("walletAddress")
      );

      statusDiv.innerHTML = `
        <div class="success-message">
          ✅ Reward refunded successfully!<br>
          Transaction ID: ${result}
        </div>
      `;
      
      refundButton.style.display = 'none';
      await window.refreshInbox?.();

    } catch (error) {
      console.error('Refund error:', error);
      statusDiv.innerHTML = `
        <div class="error-message">
          ❌ Failed to refund reward: ${error.message}
        </div>
      `;
      refundButton.disabled = false;
      refundButton.textContent = 'Try Again';
    }
  }

  // --------------------------
  // Utility Methods
  // --------------------------
  toggleShareDropdown(event) {
    event.stopPropagation();
    const menu = event.target.nextElementSibling;
    menu.style.display = menu.style.display === 'none' ? 'block' : 'none';
  }

  createMailtoLink(txid, content, type) {
    const subject = `BSV Message: ${txid}`;
    const body = type.startsWith('image/') || type.startsWith('video/') || type.startsWith('audio/') 
      ? `Blockchain Transaction ID: ${txid}\n\nThis message contains ${type} content which can be viewed at:\nhttps://whatsonchain.com/tx/${txid}`
      : `Blockchain Transaction ID: ${txid}\n\nMessage Content:\n${content}\n\nView on blockchain:\nhttps://whatsonchain.com/tx/${txid}`;

    return {
      mailto: `mailto:?subject=${encodeURIComponent(subject)}&body=${encodeURIComponent(body)}`,
      gmail: `https://mail.google.com/mail/?view=cm&fs=1&to=&su=${encodeURIComponent(subject)}&body=${encodeURIComponent(body)}`,
      outlook: `https://outlook.live.com/mail/0/deeplink/compose?subject=${encodeURIComponent(subject)}&body=${encodeURIComponent(body)}`,
      yahoo: `http://compose.mail.yahoo.com/?subject=${encodeURIComponent(subject)}&body=${encodeURIComponent(body)}`,
      content: body
    };
  }

  copyToClipboard(text) {
    navigator.clipboard.writeText(text)
      .then(() => alert('Share link copied to clipboard!'))
      .catch(error => alert('Failed to copy link: ' + error.message));
  }

  formatContent(content) {
    try {
      if (content.type.startsWith('image/')) {
        return `<div style="max-width: 100%; text-align: center;">
                  <img src="data:${content.type};base64,${content.data}" alt="Message Image" onerror="console.error('Image load failed:', this.src)" />
                </div>`;
      }
      
      if (content.type.startsWith('video/')) {
        return `<div style="max-width: 100%; text-align: center;">
                  <video controls style="max-width: 100%; height: auto;" onerror="console.error('Video load failed')">
                    <source src="data:${content.type};base64,${content.data}" type="${content.type}">
                    Your browser does not support video playback.
                  </video>
                </div>`;
      }
      
      if (content.type.startsWith('audio/')) {
        return `<div style="width: 100%; padding: 20px 0;">
                  <audio controls style="width: 100%;" onerror="console.error('Audio load failed')">
                    <source src="data:${content.type};base64,${content.data}" type="${content.type}">
                    Your browser does not support audio playback.
                  </audio>
                </div>`;
      }
      
      if (content.type === 'text/html') {
        return `<iframe class="iframe-container" sandbox="allow-scripts allow-same-origin" srcdoc="${content.data.replace(/"/g, '&quot;')}"></iframe>`;
      }
      
      if (content.type === 'application/pdf') {
        return `<iframe class="iframe-container" src="data:${content.type};base64,${content.data}"></iframe>`;
      }
      
      if (content.type.startsWith('text/')) {
        const text = content.data
          .replace(/&/g, '&amp;')
          .replace(/</g, '&lt;')
          .replace(/>/g, '&gt;')
          .replace(/"/g, '&quot;')
          .replace(/'/g, '&#039;')
          .replace(/(https?:\/\/[^\s]+)/g, '<a href="$1" target="_blank">$1</a>');
        return `<pre>${text}</pre>`;
      }
      
      return `<pre>${content.data}</pre>`;
    } catch (error) {
      console.error('Content formatting error:', error);
      return `<div class="error-message">Error displaying content: ${error.message}</div>`;
    }
  }
}

// Global wrapper functions for ease of access
function showSentMessage(txid) {
  window.messageHandler.openMessage(txid);
}

function showSent() {
  window.messageHandler.updateSentFolder();
}

// Initialize the MessageHandler
window.messageHandler = new MessageHandler();

"}]}
https://whatsonchain.com/tx/0db1a4af1ccd79d6cc9db12d26a53e5748a4205c19d6c7cfa58a0c714a2c842e