写一个小小小工具

众所周知,始皇的 ChatGPT (oaifree.com) shared站点很好用。手一抖可能就刷新了,聊天记录也就丢失了。虽然也可以导出聊天记录。但是是json格式的。不太容易阅读。我参照了之前佬友的代码。写了一个小界面。便于预览和下载。有时候GPT的回答还是很有价值的,存下来作为素材。

真的很菜,没有什么技术含量。如果对大家有用的话,欢迎star :smiling_face_with_three_hearts:

  • 蹲一个大佬把下载和预览结合起来。

感谢佬友 卡尔 · 马克思的优化

  • 代码可见评论区
19 个赞

感谢佬友贡献

2 个赞

佬友怎么裸着ip就上来了

3 个赞

服务器没备案,好麻烦 :sweat_smile:

1 个赞

我一看这个网址就有点像,再一看我平时打开html文件的时候就更像了tieba_016

2 个赞

#OpenAI添加

挺好的,不过用浏览器插件可以直接保存md格式的

1 个赞

0347AC8B
哪个插件

感谢你的分享。

受楼主启发,我也写了个cloudflare worker版的。
大家可以在这里体验:
aiuuo.com

再次感谢楼主。
由于是纯前端实现,根本不走后端请求。所以可以把html抠出来自己用。

代码

export default {
  async fetch(request, env, ctx) {
    const html = `
<!DOCTYPE html>
<html lang="zh-CN-Hans">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <link rel="icon" href="" type="image/x-icon">
  <title>简易解析ChatGPT的对话JSON</title>
  <style>
    html {font-size: 16px;}
    @media (max-width: 600px) {html {font-size: 14px;}}
    body {margin: 0; padding: 0; font-family: Arial, sans-serif; background-color: #f5f5f5;}
    .container {
      display: flex; flex-direction: column; align-items: center;
      width: 100%; max-width: 800px; margin: 0 auto; padding: 2rem;
      transition: padding 0.5s ease;
    }
    .container.loaded {align-items: flex-start; padding: 1rem;}
    #title {
      font-size: 2rem; text-align: center; margin-bottom: 2rem;
      transition: all 0.5s ease;
    }
    .container.loaded #title {
      font-size: 1.5rem; text-align: left; margin-bottom: 1rem;
    }
    .file-input-container {
      display: flex; justify-content: center; align-items: center; margin-bottom: 2rem;
      transition: opacity 0.5s ease;
    }
    .container.loaded .file-input-container {opacity: 0; height: 0; overflow: hidden; margin-bottom: 0;}
    .file-input {
      padding: 0.5rem 1rem; font-size: 1rem; cursor: pointer; border: 1px solid #ccc; border-radius: 0.25rem;
      background-color: #f5f5f5; color: #333; transition: background-color 0.3s ease;
    }
    .file-input:hover {background-color: #e0e0e0;}
    .message {
      position: relative; background-color: rgba(255, 255, 255, 0.9); border-radius: 0.5rem; padding: 1rem; margin-bottom: 1rem;
      box-shadow: 0 0 0.5rem rgba(0, 0, 0, 0.05); word-wrap: break-word;
      overflow: hidden;
      transition: max-height 0.3s ease;
    }
    .system {background-color: rgba(255, 255, 255, 0.9);}
    .user {background-color: rgba(255, 224, 178, 0.3);}
    .tool {background-color: rgba(220, 237, 200, 0.3); cursor: pointer;}
    .label {font-weight: bold; font-size: 1.1rem;}
    .header {display: flex; justify-content: space-between; align-items: center; margin-bottom: 0.5rem;}
    .timestamp {font-size: 0.875rem; color: #666;}
    .content {margin-top: 0.5rem; transition: max-height 0.3s ease;}
    .collapsed .content {max-height: 6rem; overflow: hidden;}
    .message.collapsed::after {
      content: ''; position: absolute; bottom: 0; left: 0; width: 100%; height: 50%;
      background: linear-gradient(to bottom, rgba(255,255,255,0), rgba(255,255,255,0.9));
      pointer-events: none;
      border-bottom-left-radius: 0.5rem;
      border-bottom-right-radius: 0.5rem;
    }
    pre {background-color: #f0f0f0; padding: 1rem; border-radius: 0.5rem; overflow-x: auto; white-space: pre-wrap; word-wrap: break-word; position: relative;}
    .copy-button {
      position: absolute; top: 0.5rem; right: 0.5rem; padding: 0.2rem 0.5rem; font-size: 0.875rem;
      background-color: #e0e0e0; border: none; border-radius: 0.25rem; cursor: pointer;
      transition: background-color 0.3s ease, transform 0.3s ease;
      z-index: 1;
    }
    .copy-button:hover {background-color: #d5d5d5;}
    .copy-button.copied {
      background-color: #a5d6a7;
      transform: scale(1.1);
    }
    #title-container {
      display: flex; justify-content: center; align-items: center; width: 100%;
      transition: all 0.5s ease;
    }
    .hidden {display: none;}
  </style>
</head>
<body>
  <div class="container" id="container">
    <div id="title-container">
      <h1 id="title">简易解析ChatGPT的对话JSON</h1>
    </div>
    <div class="file-input-container">
      <input type="file" id="fileInput" class="file-input" accept=".json" />
    </div>
    <div id="content"></div>
  </div>
  <script defer src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
  <script defer>
    function formatTimestamp(ts) {
      const date = new Date(ts * 1000);
      const y = date.getFullYear();
      const m = String(date.getMonth() + 1).padStart(2, '0');
      const d = String(date.getDate()).padStart(2, '0');
      const hh = String(date.getHours()).padStart(2, '0');
      const mm = String(date.getMinutes()).padStart(2, '0');
      const ss = String(date.getSeconds()).padStart(2, '0');
      return \`\${y}年\${m}月\${d}日 \${hh}:\${mm}:\${ss}\`;
    }

    function addCopyButtons() {
      const pres = document.querySelectorAll('pre');
      pres.forEach(pre => {
        const code = pre.querySelector('code');
        if (code) {
          const button = document.createElement('button');
          button.classList.add('copy-button');
          button.textContent = 'Copy';
          button.addEventListener('click', function(event) {
            event.stopPropagation();
            navigator.clipboard.writeText(code.textContent).then(() => {
              button.textContent = 'Copied!';
              button.classList.add('copied');
              setTimeout(() => {
                button.textContent = 'Copy';
                button.classList.remove('copied');
              }, 2000);
            });
          });
          pre.appendChild(button);
        }
      });
    }

    document.getElementById('fileInput').addEventListener('change', function(event) {
      const file = event.target.files[0];
      if (!file) return;
      const reader = new FileReader();
      reader.onload = function(e) {
        try {
          const json = JSON.parse(e.target.result);
          const messages = json.messages || [];
          const contentDiv = document.getElementById('content');
          contentDiv.innerHTML = '';
          messages.forEach(msg => {
            const role = msg.author.role;
            const parts = msg.content.parts || [];
            const createTime = msg.create_time;
            const timeStr = createTime ? formatTimestamp(createTime) : '';
            parts.forEach(part => {
              if (part.trim() === '') return;
              const messageDiv = document.createElement('div');
              messageDiv.classList.add('message');
              let label = '';
              if (role === 'system' || role === 'assistant') {messageDiv.classList.add('system'); label = 'ChatGPT';}
              else if (role === 'user') {messageDiv.classList.add('user'); label = 'User';}
              else if (role === 'tool') {messageDiv.classList.add('tool'); label = '(思考 ChatGPT)';}
              const headerDiv = document.createElement('div');
              headerDiv.classList.add('header');
              const labelDiv = document.createElement('div');
              labelDiv.classList.add('label');
              labelDiv.textContent = label;
              const timeDiv = document.createElement('div');
              timeDiv.classList.add('timestamp');
              timeDiv.textContent = timeStr;
              headerDiv.appendChild(labelDiv);
              headerDiv.appendChild(timeDiv);
              const contentDivInner = document.createElement('div');
              contentDivInner.classList.add('content');
              contentDivInner.innerHTML = marked.parse(part);
              messageDiv.appendChild(headerDiv);
              messageDiv.appendChild(contentDivInner);
              if (role === 'tool') {
                messageDiv.classList.add('collapsed');
                messageDiv.addEventListener('click', function() {
                  if (messageDiv.classList.contains('collapsed')) {
                    const contentHeight = contentDivInner.scrollHeight;
                    contentDivInner.style.maxHeight = contentHeight + 'px';
                    messageDiv.classList.remove('collapsed');
                    contentDivInner.addEventListener('transitionend', function handler() {
                      messageDiv.style.setProperty('--mask-display', 'none');
                      messageDiv.removeEventListener('transitionend', handler);
                    });
                  } else {
                    const currentHeight = contentDivInner.scrollHeight;
                    contentDivInner.style.maxHeight = currentHeight + 'px';
                    void contentDivInner.offsetHeight;
                    contentDivInner.style.maxHeight = '6rem';
                    messageDiv.classList.add('collapsed');
                  }
                });
              }
              contentDiv.appendChild(messageDiv);
            });
          });
          const container = document.getElementById('container');
          container.classList.add('loaded');
          addCopyButtons();
        } catch (err) {
          alert('无效的JSON文件');
        }
      };
      reader.readAsText(file);

      const title = document.getElementById('title');
      const titleContainer = document.getElementById('title-container');
      titleContainer.style.transition = 'all 0.5s ease';
      titleContainer.style.display = 'flex';
      titleContainer.style.justifyContent = 'center';
      titleContainer.style.alignItems = 'center';
      setTimeout(() => {
        titleContainer.style.justifyContent = 'flex-start';
      }, 10);
    });
  </script>
</body>
</html>`;
    return new Response(html, {
      headers: { 'Content-Type': 'text/html' },
    });
  },
};

2 个赞

纯前端为什么不部署到 pages,省得特殊字符还得转义 tieba_125

1 个赞

单纯是懒得建个仓库 :grinning:

1 个赞

可以保存到本地文件然后上传,pages 修改起来是比 worker 麻烦些 tieba_022

1 个赞

又改了一下,加了代码染色

1 个赞

感谢佬友的优化,真的优化的好好,他竟然还加了代码复制 :smiling_face_with_three_hearts:

1 个赞

先收藏了,感谢分享大佬厉害啊!

1 个赞