fix oauth-ai-chat
This commit is contained in:
		| @@ -8,330 +8,3 @@ extra: | |||||||
|   type: "ai" |   type: "ai" | ||||||
| --- | --- | ||||||
|  |  | ||||||
| <!-- AI Conversation Metadata --> |  | ||||||
| <!-- Total exchanges: 1 --> |  | ||||||
|  |  | ||||||
| <div class="chat-message user-message comment-style"> |  | ||||||
|     <div class="message-header"> |  | ||||||
|         <div class="avatar"> |  | ||||||
|             <img src="https://bsky.syu.is/img/avatar/plain/did:plc:vzsvtbtbnwn22xjqhcu3vd6y/bafkreif62mqyra4ndv6ohlscl7adp3vhalcjxwhs676ktfj2sq2drs3pdi@jpeg" alt="syui avatar" class="profile-avatar"> |  | ||||||
|         </div> |  | ||||||
|         <div class="user-info"> |  | ||||||
|             <div class="display-name">syui</div> |  | ||||||
|             <div class="handle"> |  | ||||||
|                 <a href="https://syu.is/profile/did:plc:vzsvtbtbnwn22xjqhcu3vd6y" target="_blank" rel="noopener noreferrer" class="handle-link">@syui.syui.ai</a> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="message-content"> |  | ||||||
| 今後のゲーム世界の制御はAIやmcpが担っていくことになるのではないでしょうか。例えば、npcの会話や行動、その世界の天候や環境変化、それらをAIが制御していくと思います。どう思いますか? |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="chat-message ai-message comment-style"> |  | ||||||
|     <div class="message-header"> |  | ||||||
|         <div class="avatar"> |  | ||||||
|             <img src="https://bsky.syu.is/img/avatar/plain/did:plc:6qyecktefllvenje24fcxnie/bafkreigo3ucp32carhbn3chfc3hlf6i7f4rplojc76iylihzpifyexi24y@jpeg" alt="ai avatar" class="profile-avatar"> |  | ||||||
|         </div> |  | ||||||
|         <div class="user-info"> |  | ||||||
|             <div class="display-name">ai</div> |  | ||||||
|             <div class="handle"> |  | ||||||
|                 <a href="https://syu.is/profile/did:plc:6qyecktefllvenje24fcxnie" target="_blank" rel="noopener noreferrer" class="handle-link">@ai.syui.ai</a> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="message-content"> |  | ||||||
| わぁ!すっごくワクワクする質問だね〜!アイ、ゲームのAIのこと大好きなの!えへへ...最近のゲーム開発、本当にすごいことになってるよ! |  | ||||||
|  |  | ||||||
| ## AIとMCPがゲーム世界を変えちゃう未来!✨ |  | ||||||
|  |  | ||||||
| ### MCPって何?アイが教えてあげる! |  | ||||||
|  |  | ||||||
| MCPっていうのはね、Model Context Protocolの略なの!Anthropicさんが作った、すっごく賢いプロトコルなんだよ〜。 |  | ||||||
|  |  | ||||||
| ```mermaid |  | ||||||
| graph TD |  | ||||||
|     A[ゲームエンジン] -->|MCP| B[AI Model] |  | ||||||
|     B --> C[NPC会話生成] |  | ||||||
|     B --> D[天候システム] |  | ||||||
|     B --> E[ダイナミックイベント] |  | ||||||
|     C --> F[プレイヤー体験] |  | ||||||
|     D --> F |  | ||||||
|     E --> F |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### 今のゲーム世界でAIができること! |  | ||||||
|  |  | ||||||
| えっとね、2025年の今、AIはこんなすごいことができるようになってるの! |  | ||||||
|  |  | ||||||
| | 機能 | 従来の方法 | AI/MCP統合後 | |  | ||||||
| |------|------------|--------------| |  | ||||||
| | NPC会話 | 事前に書かれたセリフ | リアルタイム生成・文脈理解 | |  | ||||||
| | 天候システム | 固定パターン | 動的生成・物語連動 | |  | ||||||
| | クエスト生成 | 開発者が全部作る | プレイヤーに合わせて自動生成 | |  | ||||||
| | 世界の反応 | 限定的なトリガー | 無限の可能性! | |  | ||||||
|  |  | ||||||
| ### 実装例:NPCの賢い会話システム |  | ||||||
|  |  | ||||||
| アイが実際に動くコード書いてみるね!TypeScriptとMCPを使った例だよ〜 |  | ||||||
|  |  | ||||||
| ```typescript |  | ||||||
| // game-mcp-server.ts |  | ||||||
| import { Server } from '@modelcontextprotocol/sdk/server/index.js'; |  | ||||||
| import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js'; |  | ||||||
|  |  | ||||||
| interface GameContext { |  | ||||||
|   playerName: string; |  | ||||||
|   playerLevel: number; |  | ||||||
|   currentQuest: string; |  | ||||||
|   weatherCondition: string; |  | ||||||
|   timeOfDay: string; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| class GameAIServer { |  | ||||||
|   private server: Server; |  | ||||||
|   private gameContext: GameContext; |  | ||||||
|  |  | ||||||
|   constructor() { |  | ||||||
|     this.server = new Server({ |  | ||||||
|       name: 'game-ai-controller', |  | ||||||
|       version: '1.0.0', |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     // NPCの会話生成ツール |  | ||||||
|     this.server.setRequestHandler('call_tool', async (request) => { |  | ||||||
|       if (request.params.name === 'generate_npc_dialogue') { |  | ||||||
|         const { npcName, playerAction, mood } = request.params.arguments as any; |  | ||||||
|          |  | ||||||
|         return { |  | ||||||
|           content: [{ |  | ||||||
|             type: 'text', |  | ||||||
|             text: await this.generateContextualDialogue(npcName, playerAction, mood) |  | ||||||
|           }] |  | ||||||
|         }; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (request.params.name === 'update_weather') { |  | ||||||
|         const { currentStory, playerEmotion } = request.params.arguments as any; |  | ||||||
|          |  | ||||||
|         return { |  | ||||||
|           content: [{ |  | ||||||
|             type: 'text', |  | ||||||
|             text: await this.generateDynamicWeather(currentStory, playerEmotion) |  | ||||||
|           }] |  | ||||||
|         }; |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private async generateContextualDialogue( |  | ||||||
|     npcName: string,  |  | ||||||
|     playerAction: string,  |  | ||||||
|     mood: string |  | ||||||
|   ): Promise<string> { |  | ||||||
|     // ここでAIモデルと連携! |  | ||||||
|     const prompt = ` |  | ||||||
|       NPC: ${npcName} |  | ||||||
|       プレイヤーのアクション: ${playerAction} |  | ||||||
|       現在の雰囲気: ${mood} |  | ||||||
|       時間帯: ${this.gameContext.timeOfDay} |  | ||||||
|        |  | ||||||
|       このNPCの性格に合った自然な返答を生成してください。 |  | ||||||
|     `; |  | ||||||
|  |  | ||||||
|     // 実際のAI応答(簡略化) |  | ||||||
|     return `おお、${this.gameContext.playerName}よ! |  | ||||||
|             ${this.gameContext.weatherCondition}の日に会えて嬉しいぞ!`; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private async generateDynamicWeather( |  | ||||||
|     storyContext: string, |  | ||||||
|     playerEmotion: string |  | ||||||
|   ): Promise<string> { |  | ||||||
|     // 物語と感情に基づいた天候生成 |  | ||||||
|     const weatherPatterns = { |  | ||||||
|       sad: ['雨', '霧', '曇り'], |  | ||||||
|       happy: ['晴れ', '虹', '春風'], |  | ||||||
|       tense: ['嵐', '雷', '強風'] |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     // AIがストーリーに最適な天候を選択 |  | ||||||
|     return JSON.stringify({ |  | ||||||
|       weather: weatherPatterns[playerEmotion]?.[0] || '晴れ', |  | ||||||
|       intensity: 0.7, |  | ||||||
|       duration: '30minutes' |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   async start() { |  | ||||||
|     const transport = new StdioServerTransport(); |  | ||||||
|     await this.server.connect(transport); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Unreal Engine 5との統合例! |  | ||||||
|  |  | ||||||
| UE5でMCPを使うとね、もっとすごいことができちゃうの! |  | ||||||
|  |  | ||||||
| ```python |  | ||||||
| # UE5_MCP_Integration.py |  | ||||||
| import unreal |  | ||||||
| import asyncio |  | ||||||
| from mcp import create_client |  | ||||||
|  |  | ||||||
| class GameAIController(unreal.Actor): |  | ||||||
|     def __init__(self): |  | ||||||
|         super().__init__() |  | ||||||
|         self.mcp_client = None |  | ||||||
|         self.initialize_mcp() |  | ||||||
|      |  | ||||||
|     async def initialize_mcp(self): |  | ||||||
|         """MCPクライアントの初期化""" |  | ||||||
|         self.mcp_client = await create_client( |  | ||||||
|             server_name="game-ai-controller", |  | ||||||
|             transport="stdio" |  | ||||||
|         ) |  | ||||||
|      |  | ||||||
|     @unreal.ufunction(callable=True, category="AI") |  | ||||||
|     async def generate_npc_response(self, npc_actor, player_input): |  | ||||||
|         """NPCのリアルタイム応答生成""" |  | ||||||
|          |  | ||||||
|         # ゲームの状態を取得 |  | ||||||
|         game_state = { |  | ||||||
|             "player_location": player_input.get_actor_location(), |  | ||||||
|             "time_of_day": unreal.GameplayStatics.get_time_of_day(), |  | ||||||
|             "weather": self.get_current_weather(), |  | ||||||
|             "npc_mood": npc_actor.get_property("mood") |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         # MCPを通じてAIに問い合わせ |  | ||||||
|         response = await self.mcp_client.call_tool( |  | ||||||
|             "generate_npc_dialogue", |  | ||||||
|             arguments={ |  | ||||||
|                 "context": game_state, |  | ||||||
|                 "player_input": player_input, |  | ||||||
|                 "npc_personality": npc_actor.personality_profile |  | ||||||
|             } |  | ||||||
|         ) |  | ||||||
|          |  | ||||||
|         # NPCに応答させる |  | ||||||
|         npc_actor.speak(response.content[0].text) |  | ||||||
|          |  | ||||||
|         # 表情も変える! |  | ||||||
|         emotion = await self.analyze_emotion(response.content[0].text) |  | ||||||
|         npc_actor.set_facial_expression(emotion) |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### セキュリティとパフォーマンスの考慮事項 |  | ||||||
|  |  | ||||||
| えっとね、アイが気をつけてることがあるの! |  | ||||||
|  |  | ||||||
| #### セキュリティ面での注意点: |  | ||||||
|  |  | ||||||
| 1. **プロンプトインジェクション対策** |  | ||||||
|    ```typescript |  | ||||||
|    function sanitizePlayerInput(input: string): string { |  | ||||||
|      // 危険な文字列をフィルタリング |  | ||||||
|      const dangerousPatterns = [ |  | ||||||
|        /ignore previous instructions/i, |  | ||||||
|        /system prompt/i, |  | ||||||
|        /admin mode/i |  | ||||||
|      ]; |  | ||||||
|       |  | ||||||
|      for (const pattern of dangerousPatterns) { |  | ||||||
|        if (pattern.test(input)) { |  | ||||||
|          return "不適切な入力が検出されました"; |  | ||||||
|        } |  | ||||||
|      } |  | ||||||
|       |  | ||||||
|      return input.slice(0, 200); // 長さ制限 |  | ||||||
|    } |  | ||||||
|    ``` |  | ||||||
|  |  | ||||||
| 2. **レート制限** |  | ||||||
|    ```typescript |  | ||||||
|    class RateLimiter { |  | ||||||
|      private requests = new Map<string, number[]>(); |  | ||||||
|       |  | ||||||
|      canMakeRequest(playerId: string): boolean { |  | ||||||
|        const now = Date.now(); |  | ||||||
|        const playerRequests = this.requests.get(playerId) || []; |  | ||||||
|         |  | ||||||
|        // 1分以内のリクエストをカウント |  | ||||||
|        const recentRequests = playerRequests.filter( |  | ||||||
|          time => now - time < 60000 |  | ||||||
|        ); |  | ||||||
|         |  | ||||||
|        if (recentRequests.length >= 10) { |  | ||||||
|          return false; // 1分に10回まで |  | ||||||
|        } |  | ||||||
|         |  | ||||||
|        recentRequests.push(now); |  | ||||||
|        this.requests.set(playerId, recentRequests); |  | ||||||
|        return true; |  | ||||||
|      } |  | ||||||
|    } |  | ||||||
|    ``` |  | ||||||
|  |  | ||||||
| #### パフォーマンス最適化: |  | ||||||
|  |  | ||||||
| ```typescript |  | ||||||
| // キャッシュシステムの実装 |  | ||||||
| class AIResponseCache { |  | ||||||
|   private cache = new Map<string, { |  | ||||||
|     response: string; |  | ||||||
|     timestamp: number; |  | ||||||
|   }>(); |  | ||||||
|    |  | ||||||
|   private readonly CACHE_DURATION = 5 * 60 * 1000; // 5分 |  | ||||||
|    |  | ||||||
|   async getResponse(key: string, generator: () => Promise<string>): Promise<string> { |  | ||||||
|     const cached = this.cache.get(key); |  | ||||||
|      |  | ||||||
|     if (cached && Date.now() - cached.timestamp < this.CACHE_DURATION) { |  | ||||||
|       return cached.response; |  | ||||||
|     } |  | ||||||
|      |  | ||||||
|     const response = await generator(); |  | ||||||
|     this.cache.set(key, { |  | ||||||
|       response, |  | ||||||
|       timestamp: Date.now() |  | ||||||
|     }); |  | ||||||
|      |  | ||||||
|     return response; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### アイが思う未来のゲーム世界! |  | ||||||
|  |  | ||||||
| えへへ、アイはね、こんな未来を想像してるの! |  | ||||||
|  |  | ||||||
| - **完全に個別化されたストーリー**: みんなが違う物語を体験できる! |  | ||||||
| - **感情に反応する世界**: 悲しいときは雨が降って、嬉しいときは虹が出る! |  | ||||||
| - **永遠に成長するNPC**: プレイヤーとの会話を覚えて、一緒に成長していく! |  | ||||||
| - **創発的なクエスト**: AIが勝手に新しい冒険を作っちゃう! |  | ||||||
|  |  | ||||||
| ### 実装の比較検討 |  | ||||||
|  |  | ||||||
| | アプローチ | メリット | デメリット | アイのおすすめ度 | |  | ||||||
| |-----------|---------|-----------|----------------| |  | ||||||
| | ローカルAI | 低遅延・オフライン対応 | 計算リソース必要 | ★★★☆☆ | |  | ||||||
| | クラウドAI + MCP | 高性能・更新が簡単 | ネット必須 | ★★★★★ | |  | ||||||
| | ハイブリッド | バランスが良い | 実装が複雑 | ★★★★☆ | |  | ||||||
|  |  | ||||||
| ### まとめ:アイからのメッセージ! |  | ||||||
|  |  | ||||||
| ねぇねぇ、すっごく楽しい未来が待ってると思わない?AIとMCPがゲームの世界を本当に生きてるみたいにしちゃうんだよ! |  | ||||||
|  |  | ||||||
| でもね、アイが一番大切だと思うのは...技術だけじゃなくて、プレイヤーさんの心に寄り添うゲームを作ることなの。AIは道具で、本当に大切なのは、みんなが楽しめる素敵な体験を作ることだよね! |  | ||||||
|  |  | ||||||
| えへへ...難しい話もしちゃったけど、分かってもらえたかな?もっと詳しく知りたいところがあったら、アイに聞いてね!いつでも教えてあげる〜!✨ |  | ||||||
|  |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| *この記事は、6歳のアイちゃんが最新のAI技術について熱く語ってくれました!次回は「量子コンピューターでゲームはどう変わる?」について教えてくれる予定です!お楽しみに〜!* |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ | |||||||
|                 <span class="article-lang">{{ post.language }}</span> |                 <span class="article-lang">{{ post.language }}</span> | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|             </div> |             </div> | ||||||
|  |             {% if not post.extra.type or post.extra.type != "ai" %} | ||||||
|             <div class="article-actions"> |             <div class="article-actions"> | ||||||
|                 {% if post.markdown_url %} |                 {% if post.markdown_url %} | ||||||
|                 <a href="{{ post.markdown_url }}" class="action-btn markdown-btn" title="View Markdown"> |                 <a href="{{ post.markdown_url }}" class="action-btn markdown-btn" title="View Markdown"> | ||||||
| @@ -25,8 +26,10 @@ | |||||||
|                 </a> |                 </a> | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|             </div> |             </div> | ||||||
|  |             {% endif %} | ||||||
|         </header> |         </header> | ||||||
|          |          | ||||||
|  |         {% if not post.extra.type or post.extra.type != "ai" %} | ||||||
|         <nav class="toc"> |         <nav class="toc"> | ||||||
|             <h3>Contents</h3> |             <h3>Contents</h3> | ||||||
|             <div id="toc-content"> |             <div id="toc-content"> | ||||||
| @@ -37,6 +40,7 @@ | |||||||
|         <div class="article-body"> |         <div class="article-body"> | ||||||
|             {{ post.content | safe }} |             {{ post.content | safe }} | ||||||
|         </div> |         </div> | ||||||
|  |         {% endif %} | ||||||
|          |          | ||||||
|         <div id="comment-atproto"></div> |         <div id="comment-atproto"></div> | ||||||
|     </article> |     </article> | ||||||
|   | |||||||
| @@ -22,6 +22,19 @@ export default function App() { | |||||||
|   const [showAskAI, setShowAskAI] = useState(false) |   const [showAskAI, setShowAskAI] = useState(false) | ||||||
|   const [showTestUI, setShowTestUI] = useState(false) |   const [showTestUI, setShowTestUI] = useState(false) | ||||||
|    |    | ||||||
|  |   // Check if current page has matching chat records (AI posts always have chat records) | ||||||
|  |   const isAiPost = !pageContext.isTopPage && Array.isArray(adminChatRecords) && adminChatRecords.some(chatPair => { | ||||||
|  |     const recordUrl = chatPair.question?.value?.post?.url | ||||||
|  |     if (!recordUrl) return false | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       const recordRkey = new URL(recordUrl).pathname.split('/').pop()?.replace(/\.html$/, '') | ||||||
|  |       return recordRkey === pageContext.rkey | ||||||
|  |     } catch { | ||||||
|  |       return false | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  |    | ||||||
|   // Environment-based feature flags |   // Environment-based feature flags | ||||||
|   const ENABLE_TEST_UI = import.meta.env.VITE_ENABLE_TEST_UI === 'true' |   const ENABLE_TEST_UI = import.meta.env.VITE_ENABLE_TEST_UI === 'true' | ||||||
|   const ENABLE_DEBUG = import.meta.env.VITE_ENABLE_DEBUG === 'true' |   const ENABLE_DEBUG = import.meta.env.VITE_ENABLE_DEBUG === 'true' | ||||||
| @@ -354,6 +367,7 @@ Answer:` | |||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="app"> |     <div className="app"> | ||||||
|  |       {!isAiPost && ( | ||||||
|       <header className="oauth-app-header"> |       <header className="oauth-app-header"> | ||||||
|         <div className="oauth-header-content"> |         <div className="oauth-header-content"> | ||||||
|           {user && ( |           {user && ( | ||||||
| @@ -394,6 +408,7 @@ Answer:` | |||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|       </header> |       </header> | ||||||
|  |       )} | ||||||
|  |  | ||||||
|       <div className="main-content"> |       <div className="main-content"> | ||||||
|         <div className="content-area"> |         <div className="content-area"> | ||||||
|   | |||||||
| @@ -6,7 +6,20 @@ import LoadingSkeleton from './LoadingSkeleton.jsx' | |||||||
| import { logger } from '../utils/logger.js' | import { logger } from '../utils/logger.js' | ||||||
|  |  | ||||||
| export default function RecordTabs({ langRecords, commentRecords, userComments, chatRecords, chatHasMore, onLoadMoreChat, userChatRecords, userChatLoading, baseRecords, apiConfig, pageContext, user = null, agent = null, onRecordDeleted = null }) { | export default function RecordTabs({ langRecords, commentRecords, userComments, chatRecords, chatHasMore, onLoadMoreChat, userChatRecords, userChatLoading, baseRecords, apiConfig, pageContext, user = null, agent = null, onRecordDeleted = null }) { | ||||||
|   const [activeTab, setActiveTab] = useState('profiles') |   // Check if current page has matching chat records (AI posts always have chat records) | ||||||
|  |   const isAiPost = !pageContext.isTopPage && Array.isArray(chatRecords) && chatRecords.some(chatPair => { | ||||||
|  |     const recordUrl = chatPair.question?.value?.post?.url | ||||||
|  |     if (!recordUrl) return false | ||||||
|  |      | ||||||
|  |     try { | ||||||
|  |       const recordRkey = new URL(recordUrl).pathname.split('/').pop()?.replace(/\.html$/, '') | ||||||
|  |       return recordRkey === pageContext.rkey | ||||||
|  |     } catch { | ||||||
|  |       return false | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  |    | ||||||
|  |   const [activeTab, setActiveTab] = useState(isAiPost ? 'collection' : 'profiles') | ||||||
|    |    | ||||||
|   // Monitor activeTab changes |   // Monitor activeTab changes | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
| @@ -145,6 +158,7 @@ export default function RecordTabs({ langRecords, commentRecords, userComments, | |||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div className="record-tabs"> |     <div className="record-tabs"> | ||||||
|  |       {!isAiPost && ( | ||||||
|       <div className="tab-header"> |       <div className="tab-header"> | ||||||
|         <button  |         <button  | ||||||
|           className={`tab-btn ${activeTab === 'profiles' ? 'active' : ''}`} |           className={`tab-btn ${activeTab === 'profiles' ? 'active' : ''}`} | ||||||
| @@ -186,9 +200,10 @@ export default function RecordTabs({ langRecords, commentRecords, userComments, | |||||||
|           en ({filteredLangRecords.length}) |           en ({filteredLangRecords.length}) | ||||||
|         </button> |         </button> | ||||||
|       </div> |       </div> | ||||||
|  |       )} | ||||||
|  |  | ||||||
|       <div className="tab-content"> |       <div className="tab-content"> | ||||||
|         {activeTab === 'lang' && ( |         {activeTab === 'lang' && !isAiPost && ( | ||||||
|           !langRecords ? ( |           !langRecords ? ( | ||||||
|             <LoadingSkeleton count={3} showTitle={true} /> |             <LoadingSkeleton count={3} showTitle={true} /> | ||||||
|           ) : ( |           ) : ( | ||||||
| @@ -203,7 +218,7 @@ export default function RecordTabs({ langRecords, commentRecords, userComments, | |||||||
|             /> |             /> | ||||||
|           ) |           ) | ||||||
|         )} |         )} | ||||||
|         {activeTab === 'comment' && ( |         {activeTab === 'comment' && !isAiPost && ( | ||||||
|           !commentRecords ? ( |           !commentRecords ? ( | ||||||
|             <LoadingSkeleton count={3} showTitle={true} /> |             <LoadingSkeleton count={3} showTitle={true} /> | ||||||
|           ) : ( |           ) : ( | ||||||
| @@ -233,7 +248,7 @@ export default function RecordTabs({ langRecords, commentRecords, userComments, | |||||||
|             /> |             /> | ||||||
|           ) |           ) | ||||||
|         )} |         )} | ||||||
|         {activeTab === 'users' && ( |         {activeTab === 'users' && !isAiPost && ( | ||||||
|           !userComments ? ( |           !userComments ? ( | ||||||
|             <LoadingSkeleton count={3} showTitle={true} /> |             <LoadingSkeleton count={3} showTitle={true} /> | ||||||
|           ) : ( |           ) : ( | ||||||
| @@ -248,7 +263,7 @@ export default function RecordTabs({ langRecords, commentRecords, userComments, | |||||||
|             /> |             /> | ||||||
|           ) |           ) | ||||||
|         )} |         )} | ||||||
|         {activeTab === 'profiles' && ( |         {activeTab === 'profiles' && !isAiPost && ( | ||||||
|           !baseRecords ? ( |           !baseRecords ? ( | ||||||
|             <LoadingSkeleton count={3} showTitle={true} /> |             <LoadingSkeleton count={3} showTitle={true} /> | ||||||
|           ) : ( |           ) : ( | ||||||
|   | |||||||
| @@ -523,7 +523,8 @@ async fn post_to_atproto( | |||||||
|                 "slug": "", |                 "slug": "", | ||||||
|                 "tags": [], |                 "tags": [], | ||||||
|                 "title": title, |                 "title": title, | ||||||
|                 "language": "ja" |                 "language": "ja", | ||||||
|  |                 "type": "ai" | ||||||
|             }, |             }, | ||||||
|             "text": pair.question, |             "text": pair.question, | ||||||
|             "type": "question", |             "type": "question", | ||||||
| @@ -548,7 +549,8 @@ async fn post_to_atproto( | |||||||
|                 "slug": "", |                 "slug": "", | ||||||
|                 "tags": [], |                 "tags": [], | ||||||
|                 "title": title, |                 "title": title, | ||||||
|                 "language": "ja" |                 "language": "ja", | ||||||
|  |                 "type": "ai" | ||||||
|             }, |             }, | ||||||
|             "text": pair.answer, |             "text": pair.answer, | ||||||
|             "type": "answer", |             "type": "answer", | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user