Transaction

9e4d6c343c6b8e7eaee03091c9fe46911bf5b998ad49b4e1faaf4e603d8aeb93
Timestamp (utc)
2025-05-28 03:54:04
Fee Paid
0.00000478 BSV
(
0.00001999 BSV
-
0.00001521 BSV
)
Fee Rate
9.982 sat/KB
Version
1
Confirmations
38,872
Size Stats
47,884 B

2 Outputs

Total Output:
0.00001521 BSV
  • jM=º{"name":"ChainDisk","timestamp":1748403571302,"files":[{"name":"NoticeBoardModuleB2.html","size":35655,"type":"text/html","content":"// COMPLETE NOTICEBOARD INTEGRATION - FINAL VERSION
// Copy this entire code and add it to your HTML after your existing noticeboard script

(function() {
  'use strict';
  
  console.log('🚀 Loading Complete Noticeboard Integration...');
  
  // Store references to original functions
  let originalDisplayMessage = null;
  
  // Complete Noticeboard Integration System
  const CompleteNoticeboardIntegration = {
    
    /**
     * Main initialization function
     */
    initialize() {
      console.log('🔧 Initializing complete noticeboard integration...');
      
      this.hookDisplayMessage();
      this.enhanceExistingNoticeboard();
      this.setupBrowsingFunctionality();
      this.addGlobalHelpers();
      this.injectStyles();
      
      console.log('✅ Complete noticeboard integration active!');
      console.log('💡 Available functions:');
      console.log('   - browseAllCommunityPosts()');
      console.log('   - checkForNoticeboardResponses()');
      console.log('   - testNoticeboardDisplay()');
    },
    
    /**
     * Hook into the existing displayMessage function
     */
    hookDisplayMessage() {
      const attemptHook = () => {
        if (typeof window.displayMessage === 'function') {
          originalDisplayMessage = window.displayMessage;
          console.log('✅ Found and hooked displayMessage function');
          
          window.displayMessage = (message) => {
            console.log('📧 Processing message:', message);
            
            if (this.isNoticeboardResponse(message)) {
              console.log('🎯 Detected noticeboard response!');
              this.handleNoticeboardResponse(message);
              return;
            }
            
            if (originalDisplayMessage) {
              originalDisplayMessage(message);
            }
          };
          
        } else {
          console.log('⏳ displayMessage not found, retrying in 2 seconds...');
          setTimeout(attemptHook, 2000);
        }
      };
      
      attemptHook();
    },
    
    /**
     * Enhance existing noticeboard functionality
     */
    enhanceExistingNoticeboard() {
      const attemptEnhance = () => {
        if (typeof window.GatewayNoticeboard !== 'undefined') {
          console.log('✅ Found existing GatewayNoticeboard, enhancing...');
          this.enhanceSearchFunctionality();
        } else {
          console.log('⏳ GatewayNoticeboard not found, retrying...');
          setTimeout(attemptEnhance, 2000);
        }
      };
      
      attemptEnhance();
    },
    
    /**
     * Enhance search functionality with real response monitoring
     */
    enhanceSearchFunctionality() {
      if (!window.GatewayNoticeboard) return;
      
      const originalSubmitSearch = window.GatewayNoticeboard.submitSearch;
      
      window.GatewayNoticeboard.submitSearch = async function(event) {
        event.preventDefault();
        
        const queryElement = document.getElementById('searchQuery');
        const query = queryElement ? queryElement.value.trim() : '';
        
        if (!query) {
          alert('Please enter what you\'re looking for');
          return;
        }
        
        if (confirm(`Search for "${query}" in Community Noticeboard?\n\nCost: 650 satoshis`)) {
          CompleteNoticeboardIntegration.showSearchLoading(query);
          
          try {
            // Call original search to send transaction
            await originalSubmitSearch.call(this, event);
            CompleteNoticeboardIntegration.monitorForSearchResponse(query);
          } catch (error) {
            console.error('❌ Search request failed:', error);
            alert('Failed to search noticeboard: ' + error.message);
          }
        }
      };
    },
    
    /**
     * Check if message is from noticeboard service
     */
    isNoticeboardResponse(message) {
      return message && 
             message.senderName === "BSV Noticeboard Service" &&
             message.protocol === "BSV_WALLETMAIL_1.0" &&
             message.messageType === "application/json";
    },
    
    /**
     * Handle noticeboard response message
     */
    handleNoticeboardResponse(message) {
      try {
        const decodedContent = atob(message.messageContent);
        const responseData = JSON.parse(decodedContent);
        
        console.log('📋 Parsed noticeboard data:', responseData);
        
        if (responseData.query && Array.isArray(responseData.results)) {
          this.displaySearchResults(responseData);
        } else if (responseData.status === "posted") {
          this.displayPostConfirmation(responseData);
        } else {
          this.displayGenericResponse(responseData);
        }
        
      } catch (error) {
        console.error('❌ Error handling noticeboard response:', error);
        if (originalDisplayMessage) {
          originalDisplayMessage(message);
        }
      }
    },
    
    /**
     * Display search results in message area
     */
    displaySearchResults(responseData) {
      const { query, total_found, results } = responseData;
      const messageContainer = this.findMessageContainer();
      
      if (!messageContainer) {
        console.error('❌ Could not find message container');
        alert(`Search results for "${query}": ${total_found} results found`);
        return;
      }
      
      let resultHTML = '';
      
      if (total_found === 0) {
        resultHTML = `
          <div class="noticeboard-message no-results">
            <div class="noticeboard-header">
              <h3>🔍 Community Search Results</h3>
              <span class="search-query">No results for "${this.escapeHtml(query)}"</span>
            </div>
            <div class="no-results-content">
              <div class="empty-icon">📭</div>
              <h4>No posts found</h4>
              <p>No community posts match your search. Try different keywords or be the first to post!</p>
              <div class="action-buttons">
                <button onclick="CompleteNoticeboardIntegration.openNoticeboard('search')" class="btn-primary">
                  🔍 Search Again
                </button>
                <button onclick="CompleteNoticeboardIntegration.openNoticeboard('post')" class="btn-secondary">
                  📝 Post First One
                </button>
              </div>
            </div>
          </div>
        `;
      } else {
        const resultsHTML = results.map((post, index) => `
          <div class="community-post" style="animation: slideIn 0.3s ease ${index * 0.1}s both;">
            <div class="post-header">
              <h4 class="post-title">${this.escapeHtml(post.title)}</h4>
              <div class="post-categories">
                ${post.categories?.map(cat => `<span class="category-tag">${this.escapeHtml(cat)}</span>`).join('') || ''}
              </div>
            </div>
            <div class="post-content">${this.escapeHtml(post.content)}</div>
            <div class="post-meta">
              <div class="meta-left">
                <span class="post-date">📅 ${this.formatDate(post.posted_date)}</span>
                <span class="post-id">ID: ${post.txid.slice(0, 8)}...</span>
              </div>
              <div class="meta-right">
                <a href="https://whatsonchain.com/tx/${post.txid}" target="_blank" class="blockchain-link">
                  🔗 Verify
                </a>
                <button onclick="CompleteNoticeboardIntegration.contactPoster('${post.txid}')" class="contact-btn">
                  💬 Contact
                </button>
              </div>
            </div>
          </div>
        `).join('');
        
        resultHTML = `
          <div class="noticeboard-message search-results">
            <div class="noticeboard-header">
              <div class="header-left">
                <h3>🎉 Community Search Results</h3>
                <div class="result-summary">
                  <span class="search-query">Search: "${this.escapeHtml(query)}"</span>
                  <span class="search-time">${new Date().toLocaleTimeString()}</span>
                </div>
              </div>
              <div class="header-right">
                <span class="result-count">${total_found} result${total_found === 1 ? '' : 's'}</span>
              </div>
            </div>
            <div class="results-container">
              ${resultsHTML}
            </div>
            <div class="action-buttons">
              <button onclick="CompleteNoticeboardIntegration.openNoticeboard('search')" class="btn-primary">
                🔍 New Search
              </button>
              <button onclick="CompleteNoticeboardIntegration.openNoticeboard('post')" class="btn-secondary">
                📝 Post Something
              </button>
              <button onclick="CompleteNoticeboardIntegration.browseAllPosts()" class="btn-tertiary">
                📋 Browse All
              </button>
            </div>
          </div>
        `;
      }
      
      messageContainer.innerHTML = resultHTML;
      messageContainer.scrollIntoView({ behavior: 'smooth' });
    },
    
    /**
     * Display post confirmation
     */
    displayPostConfirmation(responseData) {
      const messageContainer = this.findMessageContainer();
      if (!messageContainer) return;
      
      const confirmationHTML = `
        <div class="noticeboard-message post-confirmation">
          <div class="noticeboard-header success">
            <h3>✅ Post Successfully Submitted</h3>
            <span class="status-badge">Live on Community Board</span>
          </div>
          <div class="confirmation-details">
            <div class="success-animation">🎉</div>
            <h4>Your post is now live!</h4>
            <div class="detail-grid">
              <div class="detail-row">
                <span class="detail-label">Title:</span>
                <span class="detail-value">${this.escapeHtml(responseData.title)}</span>
              </div>
              <div class="detail-row">
                <span class="detail-label">Notice ID:</span>
                <span class="detail-value">${responseData.notice_id}</span>
              </div>
              <div class="detail-row">
                <span class="detail-label">Expires:</span>
                <span class="detail-value">${this.formatDate(responseData.expires_date)}</span>
              </div>
            </div>
          </div>
          <div class="action-buttons">
            <button onclick="CompleteNoticeboardIntegration.browseAllPosts()" class="btn-primary">
              📋 View All Posts
            </button>
            <button onclick="CompleteNoticeboardIntegration.openNoticeboard('post')" class="btn-secondary">
              📝 Post Another
            </button>
          </div>
        </div>
      `;
      
      messageContainer.innerHTML = confirmationHTML;
      messageContainer.scrollIntoView({ behavior: 'smooth' });
    },
    
    /**
     * Browse all posts from message history
     */
    browseAllPosts() {
      console.log('🔍 Browsing all community posts from message history...');
      
      const allPosts = this.findAllCommunityPosts();
      
      if (allPosts.length === 0) {
        this.displayBrowseMessage('No community posts found in your message history. Try searching to discover posts!');
      } else {
        this.displayAllPosts(allPosts);
      }
    },
    
    /**
     * Find all community posts in message history
     */
    findAllCommunityPosts() {
      const allPosts = [];
      
      try {
        const messageSources = [
          window.messageList,
          window.messages,
          window.recentMessages,
          this.getStoredMessages()
        ];
        
        for (const source of messageSources) {
          if (Array.isArray(source)) {
            for (const message of source) {
              if (this.isNoticeboardResponse(message)) {
                try {
                  const decodedContent = atob(message.messageContent);
                  const responseData = JSON.parse(decodedContent);
                  
                  if (responseData.results && Array.isArray(responseData.results)) {
                    responseData.results.forEach(post => {
                      post.searchQuery = responseData.query;
                      post.searchDate = message.timestamp || Date.now();
                    });
                    
                    allPosts.push(...responseData.results);
                  }
                } catch (e) {
                  console.log('Error parsing message:', e);
                }
              }
            }
          }
        }
        
        // Remove duplicates and sort
        const uniquePosts = allPosts.filter((post, index, self) => 
          index === self.findIndex(p => p.txid === post.txid)
        );
        
        uniquePosts.sort((a, b) => new Date(b.posted_date) - new Date(a.posted_date));
        
        console.log(`📋 Found ${uniquePosts.length} unique community posts`);
        return uniquePosts;
        
      } catch (error) {
        console.error('Error finding community posts:', error);
        return [];
      }
    },
    
    /**
     * Display all found posts
     */
    displayAllPosts(posts) {
      const messageContainer = this.findMessageContainer();
      if (!messageContainer) return;
      
      const postsHTML = posts.map((post, index) => `
        <div class="community-post" style="animation: slideIn 0.3s ease ${index * 0.05}s both;">
          <div class="post-header">
            <h4 class="post-title">${this.escapeHtml(post.title)}</h4>
            <div class="post-categories">
              ${post.categories?.map(cat => `<span class="category-tag">${this.escapeHtml(cat)}</span>`).join('') || ''}
            </div>
          </div>
          <div class="post-content">${this.escapeHtml(post.content)}</div>
          <div class="post-meta">
            <div class="meta-left">
              <span class="post-date">📅 ${this.formatDate(post.posted_date)}</span>
              ${post.searchQuery ? `<span class="search-info">Found via: "${post.searchQuery}"</span>` : ''}
            </div>
            <div class="meta-right">
              <a href="https://whatsonchain.com/tx/${post.txid}" target="_blank" class="blockchain-link">
                🔗 Verify
              </a>
              <button onclick="CompleteNoticeboardIntegration.contactPoster('${post.txid}')" class="contact-btn">
                💬 Contact
              </button>
            </div>
          </div>
        </div>
      `).join('');
      
      const browseHTML = `
        <div class="noticeboard-message browse-all">
          <div class="noticeboard-header">
            <div class="header-left">
              <h3>📋 All Community Posts</h3>
              <div class="browse-summary">
                <span>From your message history</span>
                <span>${posts.length} total post${posts.length === 1 ? '' : 's'}</span>
              </div>
            </div>
            <div class="header-right">
              <span class="result-count">${posts.length} Found</span>
            </div>
          </div>
          <div class="results-container">
            ${postsHTML}
          </div>
          <div class="action-buttons">
            <button onclick="CompleteNoticeboardIntegration.openNoticeboard('search')" class="btn-primary">
              🔍 Search Posts
            </button>
            <button onclick="CompleteNoticeboardIntegration.openNoticeboard('post')" class="btn-secondary">
              📝 Add New Post
            </button>
            <button onclick="CompleteNoticeboardIntegration.refreshPosts()" class="btn-tertiary">
              🔄 Refresh
            </button>
          </div>
        </div>
      `;
      
      messageContainer.innerHTML = browseHTML;
      messageContainer.scrollIntoView({ behavior: 'smooth' });
    },
    
    /**
     * Show search loading with progress
     */
    showSearchLoading(query) {
      const contentArea = document.getElementById('noticeboardContentArea');
      if (!contentArea) return;
      
      let seconds = 0;
      
      const updateLoading = () => {
        const progress = Math.min((seconds / 60) * 100, 100);
        
        contentArea.innerHTML = `
          <div class="search-loading">
            <div class="loading-animation">
              <div class="spinner"></div>
            </div>
            <h2>🔍 Searching for "${query}"</h2>
            <p class="loading-status">Your search request has been sent to the community sentinel...</p>
            <div class="progress-container">
              <div class="progress-bar">
                <div class="progress-fill" style="width: ${progress}%"></div>
              </div>
              <span class="progress-text">${seconds}s elapsed</span>
            </div>
            <div class="loading-steps">
              <div class="step completed">
                <span class="step-icon">✅</span>
                <span class="step-text">Transaction sent (650 sats)</span>
              </div>
              <div class="step ${seconds > 5 ? 'completed' : 'active'}">
                <span class="step-icon">${seconds > 5 ? '✅' : '⏳'}</span>
                <span class="step-text">Sentinel processing query</span>
              </div>
              <div class="step ${seconds > 15 ? 'active' : ''}">
                <span class="step-icon">⏳</span>
                <span class="step-text">Searching community database</span>
              </div>
            </div>
            <button onclick="CompleteNoticeboardIntegration.cancelSearch()" class="cancel-btn">
              Cancel Search
            </button>
          </div>
        `;
      };
      
      updateLoading();
      
      const timer = setInterval(() => {
        seconds++;
        updateLoading();
        
        if (seconds >= 60) {
          clearInterval(timer);
          this.showSearchTimeout(query);
        }
      }, 1000);
      
      this.searchTimer = timer;
    },
    
    /**
     * Monitor for search response
     */
    monitorForSearchResponse(originalQuery) {
      console.log('⏳ Monitoring for search response...');
      
      let attempts = 0;
      const maxAttempts = 60;
      let responseReceived = false;
      
      const checkForResponse = setInterval(() => {
        if (responseReceived) {
          clearInterval(checkForResponse);
          return;
        }
        
        attempts++;
        
        try {
          const response = this.checkForResponse(originalQuery);
          
          if (response) {
            responseReceived = true;
            clearInterval(checkForResponse);
            
            if (this.searchTimer) {
              clearInterval(this.searchTimer);
            }
            
            console.log('✅ Search response received!', response);
            
            setTimeout(() => {
              if (!document.querySelector('.search-results')) {
                this.displaySearchResults(response);
              }
            }, 1000);
            
            return;
          }
          
          if (attempts >= maxAttempts) {
            clearInterval(checkForResponse);
            this.showSearchTimeout(originalQuery);
          }
          
        } catch (error) {
          console.error('Error checking for response:', error);
        }
      }, 1000);
    },
    
    /**
     * Check for response in messages
     */
    checkForResponse(expectedQuery) {
      try {
        const messageSources = [window.messageList, window.messages];
        
        for (const source of messageSources) {
          if (Array.isArray(source) && source.length > 0) {
            const recentMessages = source.slice(-5);
            
            for (const message of recentMessages) {
              if (this.isNoticeboardResponse(message)) {
                try {
                  const decodedContent = atob(message.messageContent);
                  const responseData = JSON.parse(decodedContent);
                  
                  if (responseData.query && Array.isArray(responseData.results)) {
                    if (!expectedQuery || responseData.query === expectedQuery) {
                      return responseData;
                    }
                  }
                } catch (e) {
                  console.log('Error parsing message:', e);
                }
              }
            }
          }
        }
        
        return null;
      } catch (error) {
        console.error('Error checking for response:', error);
        return null;
      }
    },
    
    /**
     * Setup browsing functionality
     */
    setupBrowsingFunctionality() {
      const addBrowseButton = () => {
        if (window.NoticeboardActions) {
          window.NoticeboardActions.browseAllPosts = () => this.browseAllPosts();
          console.log('✅ Added browse functionality to NoticeboardActions');
        } else {
          setTimeout(addBrowseButton, 2000);
        }
      };
      
      addBrowseButton();
    },
    
    /**
     * Add global helper functions
     */
    addGlobalHelpers() {
      window.browseAllCommunityPosts = () => {
        const posts = this.findAllCommunityPosts();
        console.log('📋 Found community posts:', posts);
        return posts;
      };
      
      window.checkForNoticeboardResponses = () => {
        console.log('🔍 Checking for noticeboard responses...');
        const found = this.checkForResponse();
        if (found) {
          console.log('✅ Found response:', found);
          this.displaySearchResults(found);
        } else {
          console.log('❌ No responses found');
          alert('No noticeboard responses found in recent messages');
        }
      };
      
      window.testNoticeboardDisplay = () => {
        const testResponse = {
          query: "test",
          total_found: 2,
          results: [
            {
              txid: "test123abc",
              title: "Test Community Post",
              content: "This is a test post to verify the display system is working correctly.",
              posted_date: new Date().toISOString(),
              categories: ["test", "community"]
            },
            {
              txid: "test456def", 
              title: "Another Test Post",
              content: "Second test post with different content to show multiple results.",
              posted_date: new Date(Date.now() - 86400000).toISOString(),
              categories: ["test"]
            }
          ]
        };
        
        this.displaySearchResults(testResponse);
      };
    },
    
    /**
     * Utility Functions
     */
    openNoticeboard(action = null) {
      if (typeof window.showNoticeboard === 'function') {
        window.showNoticeboard();
        
        setTimeout(() => {
          if (action === 'search' && window.NoticeboardActions?.showSearchForm) {
            window.NoticeboardActions.showSearchForm();
          } else if (action === 'post' && window.NoticeboardActions?.showPostForm) {
            window.NoticeboardActions.showPostForm();
          }
        }, 500);
      } else {
        alert('Noticeboard interface not available');
      }
    },
    
    contactPoster(txid) {
      alert(`Contact functionality for post ${txid}.\n\nThis would integrate with your messaging system to contact the poster.`);
    },
    
    cancelSearch() {
      if (this.searchTimer) {
        clearInterval(this.searchTimer);
      }
      
      if (window.GatewayNoticeboard?.showWelcome) {
        window.GatewayNoticeboard.showWelcome();
      }
    },
    
    refreshPosts() {
      this.browseAllPosts();
    },
    
    displayBrowseMessage(message) {
      const messageContainer = this.findMessageContainer();
      if (!messageContainer) return;
      
      const browseHTML = `
        <div class="noticeboard-message browse-empty">
          <div class="noticeboard-header">
            <h3>📋 Community Posts</h3>
          </div>
          <div class="empty-content">
            <p>${message}</p>
            <div class="action-buttons">
              <button onclick="CompleteNoticeboardIntegration.openNoticeboard('search')" class="btn-primary">
                🔍 Search Community
              </button>
              <button onclick="CompleteNoticeboardIntegration.openNoticeboard('post')" class="btn-secondary">
                📝 Post Something
              </button>
            </div>
          </div>
        </div>
      `;
      
      messageContainer.innerHTML = browseHTML;
    },
    
    displayGenericResponse(responseData) {
      const messageContainer = this.findMessageContainer();
      if (!messageContainer) return;
      
      const genericHTML = `
        <div class="noticeboard-message generic">
          <div class="noticeboard-header">
            <h3>📋 Community Noticeboard Response</h3>
          </div>
          <div class="generic-content">
            <pre>${JSON.stringify(responseData, null, 2)}</pre>
          </div>
          <div class="action-buttons">
            <button onclick="CompleteNoticeboardIntegration.openNoticeboard()" class="btn-primary">
              📋 Open Noticeboard
            </button>
          </div>
        </div>
      `;
      
      messageContainer.innerHTML = genericHTML;
    },
    
    showSearchTimeout(query) {
      const contentArea = document.getElementById('noticeboardContentArea');
      if (!contentArea) return;
      
      contentArea.innerHTML = `
        <div class="search-timeout">
          <div class="timeout-icon">⏱️</div>
          <h3>Search Taking Longer Than Expected</h3>
          <p>Your search for "${query}" was sent successfully, but the response is delayed.</p>
          <div class="action-buttons">
            <button onclick="window.checkForNoticeboardResponses()" class="btn-primary">
              🔍 Check for Response
            </button>
            <button onclick="CompleteNoticeboardIntegration.openNoticeboard('search')" class="btn-secondary">
              🔄 Try Again
            </button>
          </div>
        </div>
      `;
    },
    
    findMessageContainer() {
      const containers = [
        '#messageContent',
        '#messageDisplay',
        '.message-content',
        '.message-display',
        '.content-area',
        '#mainContent',
        '#noticeboardContentArea'
      ];
      
      for (const selector of containers) {
        const container = document.querySelector(selector);
        if (container) {
          console.log(`📦 Using container: ${selector}`);
          return container;
        }
      }
      
      console.error('❌ Could not find message container');
      return null;
    },
    
    getStoredMessages() {
      try {
        const stored = localStorage.getItem('messages') || localStorage.getItem('messageList');
        return stored ? JSON.parse(stored) : null;
      } catch (e) {
        return null;
      }
    },
    
    escapeHtml(text) {
      const div = document.createElement('div');
      div.textContent = text;
      return div.innerHTML;
    },
    
    formatDate(dateString) {
      const date = new Date(dateString);
      const now = new Date();
      const diffTime = Math.abs(now - date);
      const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24));
      
      if (diffDays === 1) return 'Yesterday';
      if (diffDays <= 7) return `${diffDays} days ago`;
      return date.toLocaleDateString();
    },
    
    /**
     * Inject CSS styles
     */
    injectStyles() {
      if (document.getElementById('noticeboardMessageStyles')) return;
      
      const styles = document.createElement('style');
      styles.id = 'noticeboardMessageStyles';
      styles.textContent = `
        .noticeboard-message {
          background: linear-gradient(135deg, #f8fafc 0%, #e2e8f0 100%);
          border: 2px solid #4facfe;
          border-radius: 15px;
          padding: 25px;
          margin: 20px 0;
          font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
          box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
        }
        
        .noticeboard-header {
          display: flex;
          justify-content: space-between;
          align-items: center;
          margin-bottom: 20px;
          padding-bottom: 15px;
          border-bottom: 2px solid #e2e8f0;
        }
        
        .noticeboard-header.success {
          border-bottom-color: #48bb78;
        }
        
        .noticeboard-header h3 {
          margin: 0;
          color: #2d3748;
          font-size: 1.3rem;
        }
        
        .search-query {
          color: #4a5568;
          font-style: italic;
          font-size: 0.9rem;
        }
        
        .result-count, .status-badge {
          background: #4facfe;
          color: white;
          padding: 6px 12px;
          border-radius: 20px;
          font-size: 0.85rem;
          font-weight: 600;
        }
        
        .status-badge {
          background: #48bb78;
        }
        
        .results-container {
          max-height: 500px;
          overflow-y: auto;
          margin: 20px 0;
        }
        
        .community-post {
          background: white;
          border: 1px solid #e2e8f0;
          border-radius: 10px;
          padding: 20px;
          margin-bottom: 15px;
          transition: all 0.2s ease;
        }
        
        .community-post:hover {
          box-shadow: 0 4px 12px rgba(0,0,0,0.1);
          transform: translateY(-2px);
        }
        
        .post-header {
          display: flex;
          justify-content: space-between;
          align-items: flex-start;
          margin-bottom: 12px;
        }
        
        .post-title {
          margin: 0;
          color: #2d3748;
          font-size: 1.1rem;
          font-weight: 600;
        }
        
        .post-categories {
          display: flex;
          gap: 5px;
          flex-wrap: wrap;
        }
        
        .category-tag {
          background: #edf2f7;
          color: #4a5568;
          padding: 3px 8px;
          border-radius: 12px;
          font-size: 0.75rem;
          font-weight: 500;
        }
        
        .post-content {
          color: #4a5568;
          line-height: 1.6;
          margin: 12px 0;
        }
        
        .post-meta {
          display: flex;
          justify-content: space-between;
          align-items: center;
          font-size: 0.85rem;
          color: #718096;
          padding-top: 12px;
          border-top: 1px solid #f1f5f9;
          flex-wrap: wrap;
          gap: 10px;
        }
        
        .search-info {
          font-style: italic;
          color: #9ca3af;
        }
        
        .blockchain-link {
          color: #4facfe;
          text-decoration: none;
          font-weight: 500;
        }
        
        .blockchain-link:hover {
          text-decoration: underline;
        }
        
        .contact-btn {
          background: #48bb78;
          color: white;
          border: none;
          padding: 6px 12px;
          border-radius: 6px;
          font-size: 0.8rem;
          cursor: pointer;
          font-weight: 500;
        }
        
        .contact-btn:hover {
          background: #38a169;
        }
        
        .action-buttons {
          display: flex;
          gap: 12px;
          margin-top: 25px;
          padding-top: 20px;
          border-top: 1px solid #e2e8f0;
          flex-wrap: wrap;
        }
        
        .btn-primary, .btn-secondary, .btn-tertiary {
          flex: 1;
          min-width: 140px;
          padding: 12px 20px;
          border: none;
          border-radius: 8px;
          font-weight: 600;
          cursor: pointer;
          transition: all 0.2s ease;
          font-size: 0.95rem;
        }
        
        .btn-primary {
          background: #4facfe;
          color: white;
        }
        
        .btn-primary:hover {
          background: #3182ce;
          transform: translateY(-1px);
        }
        
        .btn-secondary {
          background: #48bb78;
          color: white;
        }
        
        .btn-secondary:hover {
          background: #38a169;
          transform: translateY(-1px);
        }
        
        .btn-tertiary {
          background: #718096;
          color: white;
        }
        
        .btn-tertiary:hover {
          background: #4a5568;
          transform: translateY(-1px);
        }
        
        .search-loading {
          text-align: center;
          padding: 60px 20px;
        }
        
        .spinner {
          width: 50px;
          height: 50px;
          border: 4px solid #e2e8f0;
          border-top: 4px solid #4facfe;
          border-radius: 50%;
          animation: spin 1s linear infinite;
          margin: 0 auto 30px;
        }
        
        .progress-container {
          max-width: 300px;
          margin: 20px auto;
        }
        
        .progress-bar {
          background: #e2e8f0;
          border-radius: 10px;
          height: 8px;
          overflow: hidden;
        }
        
        .progress-fill {
          background: #4facfe;
          height: 100%;
          transition: width 0.5s ease;
        }
        
        .loading-steps {
          max-width: 400px;
          margin: 30px auto;
          text-align: left;
        }
        
        .step {
          display: flex;
          align-items: center;
          margin: 15px 0;
          opacity: 0.5;
        }
        
        .step.active, .step.completed {
          opacity: 1;
        }
        
        .step-icon {
          margin-right: 10px;
          width: 20px;
        }
        
        .cancel-btn {
          background: #e53e3e;
          color: white;
          border: none;
          padding: 10px 20px;
          border-radius: 6px;
          cursor: pointer;
          margin-top: 30px;
        }
        
        @keyframes spin {
          0% { transform: rotate(0deg); }
          100% { transform: rotate(360deg); }
        }
        
        @keyframes slideIn {
          from { opacity: 0; transform: translateY(20px); }
          to { opacity: 1; transform: translateY(0); }
        }
        
        @media (max-width: 768px) {
          .noticeboard-header {
            flex-direction: column;
            gap: 10px;
            text-align: center;
          }
          
          .post-header {
            flex-direction: column;
            gap: 8px;
          }
          
          .post-meta {
            flex-direction: column;
            text-align: center;
            gap: 5px;
          }
          
          .action-buttons {
            flex-direction: column;
          }
          
          .btn-primary, .btn-secondary, .btn-tertiary {
            min-width: auto;
          }
        }
      `;
      
      document.head.appendChild(styles);
    }
  };
  
  // Initialize the integration
  CompleteNoticeboardIntegration.initialize();
  
  // Add to global scope
  window.CompleteNoticeboardIntegration = CompleteNoticeboardIntegration;
  
  console.log('✅ Complete noticeboard integration loaded!');
  console.log('🧪 Test functions available:');
  console.log('   - testNoticeboardDisplay()');
  console.log('   - browseAllCommunityPosts()');
  console.log('   - checkForNoticeboardResponses()');
  
})();"}]}
    https://whatsonchain.com/tx/9e4d6c343c6b8e7eaee03091c9fe46911bf5b998ad49b4e1faaf4e603d8aeb93