update layout

This commit is contained in:
2025-06-16 01:43:51 +09:00
parent a76933c23b
commit 60d9e8292c
7 changed files with 118 additions and 29 deletions

View File

@@ -248,7 +248,7 @@ a.view-markdown:any-link {
} }
.post-title a { .post-title a {
color: #1f2328; color: var(--theme-color);
text-decoration: none; text-decoration: none;
font-size: 18px; font-size: 18px;
font-weight: 600; font-weight: 600;

View File

@@ -20,19 +20,6 @@
<a href="{{ post.url }}">{{ post.title }}</a> <a href="{{ post.url }}">{{ post.title }}</a>
</h3> </h3>
{% if post.excerpt %}
<p class="post-excerpt">{{ post.excerpt }}</p>
{% endif %}
<div class="post-actions">
<a href="{{ post.url }}" class="read-more">Read more</a>
{% if post.markdown_url %}
<a href="{{ post.markdown_url }}" class="view-markdown" title="View Markdown">.md</a>
{% endif %}
{% if post.translation_url %}
<a href="{{ post.translation_url }}" class="view-translation" title="View Translation">🌐</a>
{% endif %}
</div>
</div> </div>
</article> </article>
{% endfor %} {% endfor %}

View File

@@ -1080,6 +1080,7 @@ function App() {
</div> </div>
</div> </div>
) : ( ) : (
/* User Section */
<div className="user-section"> <div className="user-section">
<div className="user-info"> <div className="user-info">
<div className="user-profile"> <div className="user-profile">
@@ -1187,7 +1188,6 @@ function App() {
</div> </div>
</div> </div>
)} )}
</div> </div>
)} )}
@@ -1197,25 +1197,25 @@ function App() {
className={`tab-button ${activeTab === 'comments' ? 'active' : ''}`} className={`tab-button ${activeTab === 'comments' ? 'active' : ''}`}
onClick={() => setActiveTab('comments')} onClick={() => setActiveTab('comments')}
> >
Comments ({comments.filter(shouldShowComment).length}) comment ({comments.filter(shouldShowComment).length})
</button> </button>
<button <button
className={`tab-button ${activeTab === 'ai-chat' ? 'active' : ''}`} className={`tab-button ${activeTab === 'ai-chat' ? 'active' : ''}`}
onClick={() => setActiveTab('ai-chat')} onClick={() => setActiveTab('ai-chat')}
> >
AI Chat ({aiChatHistory.length}) chat ({aiChatHistory.length})
</button> </button>
<button <button
className={`tab-button ${activeTab === 'lang-en' ? 'active' : ''}`} className={`tab-button ${activeTab === 'lang-en' ? 'active' : ''}`}
onClick={() => setActiveTab('lang-en')} onClick={() => setActiveTab('lang-en')}
> >
AI Lang:en ({langEnRecords.length}) lang:en ({langEnRecords.length})
</button> </button>
<button <button
className={`tab-button ${activeTab === 'ai-comment' ? 'active' : ''}`} className={`tab-button ${activeTab === 'ai-comment' ? 'active' : ''}`}
onClick={() => setActiveTab('ai-comment')} onClick={() => setActiveTab('ai-comment')}
> >
AI Comment ({aiCommentRecords.length}) feedback ({aiCommentRecords.length})
</button> </button>
</div> </div>
@@ -1534,11 +1534,113 @@ function App() {
</div> </div>
)} )}
{/* Show authentication status on non-post pages */} {/* User Section - moved from above */}
{user && !appConfig.rkey && ( {user && !appConfig.rkey && (
<div className="auth-status"> <div className="user-section">
<p> Authenticated as @{user.handle}</p> <div className="user-info">
<p><small>Visit a post page to comment</small></p> <div className="user-profile">
<img
src={user.avatar || generatePlaceholderAvatar(user.handle)}
alt="User Avatar"
className="user-avatar"
/>
<div className="user-details">
<h3>{user.displayName || user.handle}</h3>
<p className="user-handle">@{user.handle}</p>
<p className="user-did">{user.did}</p>
</div>
</div>
<button onClick={handleLogout} className="logout-button">
Logout
</button>
</div>
{/* Admin Section - User Management */}
{isAdmin(user) && (
<div className="admin-section">
<h3> - </h3>
{/* User List Form */}
<div className="user-list-form">
<textarea
id="user-list-input"
name="userList"
value={userListInput}
onChange={(e) => setUserListInput(e.target.value)}
placeholder="ユーザーハンドルをカンマ区切りで入力&#10;例: syui.ai, yui.syui.ai, user.bsky.social"
rows={3}
disabled={isPostingUserList}
/>
<div className="form-actions">
<span className="admin-hint"></span>
<button
onClick={handlePostUserList}
disabled={isPostingUserList || !userListInput.trim()}
className="post-button"
>
{isPostingUserList ? 'Posting...' : 'Post User List'}
</button>
</div>
</div>
{/* User List Records */}
<div className="user-list-records">
<h4>稿</h4>
{userListRecords.length === 0 ? (
<p className="no-user-lists">稿</p>
) : (
userListRecords.map((record, index) => (
<div key={index} className="user-list-item">
<div className="user-list-header">
<span className="user-list-date">
{new Date(record.value.createdAt).toLocaleString()}
</span>
<div className="user-list-actions">
<button
onClick={() => toggleJsonDisplay(record.uri)}
className="json-button"
title="Show/Hide JSON"
>
{showJsonFor === record.uri ? 'Hide' : 'JSON'}
</button>
</div>
</div>
<div className="user-list-content">
{record.value.users && record.value.users.length > 0 && (
<div className="user-handles">
{record.value.users.map((user: any, userIndex: number) => (
<span key={userIndex} className="user-handle-tag">
@{user.handle}
{user.pds && <span className="pds-info"> ({user.pds})</span>}
</span>
))}
</div>
)}
<div className="user-list-meta">
<small>URI: {record.uri}</small>
<br />
<small>Updated by: {record.value.updatedBy?.handle || 'unknown'}</small>
</div>
{/* JSON Display */}
{showJsonFor === record.uri && (
<div className="json-display">
<h5>JSON Record:</h5>
<pre className="json-content">
{JSON.stringify(record, null, 2)}
</pre>
</div>
)}
</div>
</div>
))
)}
</div>
</div>
)}
</div> </div>
)} )}
</section> </section>

View File

@@ -14,7 +14,7 @@ const response = await fetch(`${aiConfig.host}/api/generate`, {
options: { options: {
temperature: 0.9, temperature: 0.9,
top_p: 0.9, top_p: 0.9,
num_predict: 80, num_predict: 200,
repeat_penalty: 1.1, repeat_penalty: 1.1,
} }
}), }),

View File

@@ -199,7 +199,7 @@ Answer:`;
options: { options: {
temperature: 0.9, temperature: 0.9,
top_p: 0.9, top_p: 0.9,
num_predict: 80, // Shorter responses for faster generation num_predict: 200, // Longer responses for better answers
repeat_penalty: 1.1, repeat_penalty: 1.1,
} }
}), }),

View File

@@ -1,7 +1,7 @@
#!/bin/zsh #!/bin/zsh
function _env() { function _env() {
d=${0:a:h:h} d=${0:a:h}
ailog=$d/target/release/ailog ailog=$d/target/release/ailog
oauth=$d/oauth oauth=$d/oauth
myblog=$d/my-blog myblog=$d/my-blog

View File

@@ -1050,7 +1050,7 @@ async fn generate_ai_content(content: &str, prompt_type: &str, ai_config: &AiCon
}; };
format!( format!(
"{}\n\n# 指示\nこのブログ記事を読んで、アイらしい感想を一言でください。\n- 30文字以内の短い感想\n- 技術的な内容への素朴な驚きや発見\n- 「わー!」「すごい!」など、アイらしい感嘆詞で始める\n- 簡潔で分かりやすく\n\n# ブログ記事(要約)\n{}\n\n# 出力形式\n一言の感想のみ(説明や詳細は不要):", "{}\n\n# 指示\nこのブログ記事を読んで、アイらしい感想をください。\n- 100文字以内の感想\n- 技術的な内容への素朴な驚きや発見\n- 「わー!」「すごい!」など、アイらしい感嘆詞で始める\n- 簡潔で分かりやすく\n\n# ブログ記事(要約)\n{}\n\n# 出力形式\n感想のみ(説明や詳細は不要):",
system_prompt, limited_content system_prompt, limited_content
) )
}, },
@@ -1058,7 +1058,7 @@ async fn generate_ai_content(content: &str, prompt_type: &str, ai_config: &AiCon
}; };
let num_predict = match prompt_type { let num_predict = match prompt_type {
"comment" => 50, // Very short for comments (about 30-40 characters) "comment" => 150, // Longer for comments (about 100 characters)
"translate" => 3000, // Much longer for translations "translate" => 3000, // Much longer for translations
_ => 300, _ => 300,
}; };