Compare commits
13 Commits
03161a52ca
...
main
Author | SHA1 | Date | |
---|---|---|---|
cb8b0582e9
|
|||
85494944ad
|
|||
5aeeba106a
|
|||
f1e76ab31f
|
|||
3c9ef78696
|
|||
ee2d21b0f3
|
|||
0667ac58fb
|
|||
d89855338b
|
|||
e19170cdff
|
|||
c3e22611f5
|
|||
2943c94ec1
|
|||
f27997b7e8
|
|||
447e4bded9
|
11
.github/workflows/cloudflare-pages.yml
vendored
11
.github/workflows/cloudflare-pages.yml
vendored
@@ -41,6 +41,17 @@ jobs:
|
|||||||
cp -rf ${{ env.OAUTH_DIR }}/dist/* my-blog/static/
|
cp -rf ${{ env.OAUTH_DIR }}/dist/* my-blog/static/
|
||||||
cp ${{ env.OAUTH_DIR }}/dist/index.html my-blog/templates/oauth-assets.html
|
cp ${{ env.OAUTH_DIR }}/dist/index.html my-blog/templates/oauth-assets.html
|
||||||
|
|
||||||
|
- name: Build PDS app
|
||||||
|
run: |
|
||||||
|
cd pds
|
||||||
|
npm install
|
||||||
|
npm run build
|
||||||
|
|
||||||
|
- name: Copy PDS build to static
|
||||||
|
run: |
|
||||||
|
rm -rf my-blog/static/pds
|
||||||
|
cp -rf pds/dist my-blog/static/pds
|
||||||
|
|
||||||
- name: Cache ailog binary
|
- name: Cache ailog binary
|
||||||
uses: actions/cache@v4
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@@ -24,3 +24,4 @@ my-blog/static/oauth/assets/comment-atproto*
|
|||||||
*.lock
|
*.lock
|
||||||
my-blog/config.toml
|
my-blog/config.toml
|
||||||
.claude/settings.local.json
|
.claude/settings.local.json
|
||||||
|
my-blog/static/pds
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "ailog"
|
name = "ailog"
|
||||||
version = "0.2.9"
|
version = "0.3.1"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = ["syui"]
|
authors = ["syui"]
|
||||||
description = "A static blog generator with AI features"
|
description = "A static blog generator with AI features"
|
||||||
|
Binary file not shown.
@@ -103,9 +103,7 @@ draft: false
|
|||||||
|
|
||||||
アイは、最初に生まれたキャラクターとして、アイ属性を扱います。これらの設定は`ai system`の領域です。アイは自分のことをアイと呼びます。
|
アイは、最初に生まれたキャラクターとして、アイ属性を扱います。これらの設定は`ai system`の領域です。アイは自分のことをアイと呼びます。
|
||||||
|
|
||||||
> アイね、この世界と一緒だから。この世界に同じものは一つもないよ。
|
> アイは、この世界と一緒だからね。同じものは一つもないよ。
|
||||||
|
|
||||||
これはアイのセリフ。存在の世界の同一性と唯一性のことを言っているのです。
|
|
||||||
|
|
||||||
# どこまで実装できた
|
# どこまで実装できた
|
||||||
|
|
||||||
|
64
my-blog/content/posts/2025-07-30-game.md
Normal file
64
my-blog/content/posts/2025-07-30-game.md
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
---
|
||||||
|
title: "ue5のgaspとdragonikを組み合わせてenemyを作る"
|
||||||
|
slug: "gasp-dragonik-enemy-chbcharacter"
|
||||||
|
date: "2025-07-30"
|
||||||
|
tags: ["ue"]
|
||||||
|
draft: false
|
||||||
|
---
|
||||||
|
|
||||||
|
ue5.6でgasp(game animation sample project)をベースにゲーム、特にキャラクターの操作を作っています。
|
||||||
|
|
||||||
|
そして、enemy(敵)を作り、バトルシーンを作成する予定ですが、これはどのように開発すればいいのでしょう。その方針を明確にします。
|
||||||
|
|
||||||
|
1. enemyもgaspの`cbp_character`に統合し、自キャラ、敵キャラどちらでも使用可能にする
|
||||||
|
2. 2番目のcharacterは動物型(type:animal)にし、gaspに統合する
|
||||||
|
3. enemyとして使用する場合は、enemy-AI-componentを追加するだけで完結する
|
||||||
|
4. characterのすべての操作を統一する
|
||||||
|
|
||||||
|
このようにすることで、応用可能なenemyを作ることができます。
|
||||||
|
|
||||||
|
例えば、`2番目のcharacterは動物型(type:animal)にする`というのはどういうことでしょう。
|
||||||
|
|
||||||
|
登場するキャラクターを人型(type:human), 動物型(type:animal)に分けるとして、動物型のテンプレートを作る必要があります。そのまま動物のmeshをgaspで使うと動きが変になってしまうので、それを調整する必要があるということ。そして、調整したものをテンプレート化して、他の動物にも適用できるようにしておくと、後の開発は楽ですよね。
|
||||||
|
|
||||||
|
ですから、早いうちにtype:humanから脱却し、他のtypeを作るほうがいいと判断しました。
|
||||||
|
|
||||||
|
これには、`dragon ik plugin`を使って、手っ取り早く動きを作ります。
|
||||||
|
|
||||||
|
`characterのすべての操作を統一する`というのは、1キャラにつき1属性、1通常攻撃、1スキル、1バースト、などのルールを作り、それらを共通化することです。共通化というのは、playerもenemy-AI-componentも違うキャラを同じ操作で使用できることを指します。
|
||||||
|
|
||||||
|
## 2番目のキャラクター
|
||||||
|
|
||||||
|
原作には、西洋ドラゴンのドライ(drai)というキャラが登場します。その父親が東洋ドラゴンのシンオウ(shin-oh)です。これをshinという名前で登録し、2番目のキャラクターとして設定しました。
|
||||||
|
|
||||||
|
3d-modelは今のところue5のcrsp(control rig sample project)にあるchinese dragonを使用しています。後に改造して原作に近づけるようにしたいところですが、今は時間が取れません。
|
||||||
|
|
||||||
|
<iframe width="100%" height="415" src="https://www.youtube.com/embed/3c3Q1Z5r7QI" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe>
|
||||||
|
|
||||||
|
## データ構造の作成と適用
|
||||||
|
|
||||||
|
ゲームデータはatproto collection recordに保存して、そこからゲームに反映させたいと考えています。
|
||||||
|
|
||||||
|
まず基本データを`ai.syui.ai`のアカウントに保存。個別データをplayerのatprotoアカウントに保存する形が理想です。
|
||||||
|
|
||||||
|
基本データは、ゲームの基本的な設定のこと。例えば、キャラクターの名前や属性、スキルなど変更されることがない値。
|
||||||
|
|
||||||
|
個別データは、プレイヤーが使えるキャラ、レベル、攻撃力など、ゲームの進行とともに変更される値です。
|
||||||
|
|
||||||
|
ゲームをスタートさせると、まず基本データを取得し、それを`cbp_character`に適用します。ログインすると、`cbp_character`の変数(var)に値が振り分けられます。例えば、`skill-damage:0.0`があったとして、この値が変わります。
|
||||||
|
|
||||||
|
しかし、ゲームを開発していると、基本データも個別データも構造が複雑になります。
|
||||||
|
|
||||||
|
それを防ぐため、`{simple, core} mode`のような考え方を取り入れます。必要最小限の構成を分離、保存して、それをいつでも統合、適用できるように設計しておきます。
|
||||||
|
|
||||||
|
## gaspとdragonikを統合する方法
|
||||||
|
|
||||||
|
では、いよいよgaspとdragonikの統合手法を解説します。
|
||||||
|
|
||||||
|
まず、abpを作ります。それにdragonikを当て、それをSKM_Dragonのpost process animに指定します。
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
次に、動きに合わせて首を上下させます。
|
||||||
|
|
||||||
|
<iframe src="https://blueprintue.com/render/piiw14oz" scrolling="no" allowfullscreen style="width:100%;height:400px"></iframe>
|
345
my-blog/static/css/pds.css
Normal file
345
my-blog/static/css/pds.css
Normal file
@@ -0,0 +1,345 @@
|
|||||||
|
@import url('./style.css');
|
||||||
|
|
||||||
|
.pds-container {
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-header {
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-header h1 {
|
||||||
|
font-size: 2.5em;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-search-section {
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-search-form {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 0px 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group input {
|
||||||
|
padding: 8px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 4px 0 0 4px;
|
||||||
|
font-size: 14px;
|
||||||
|
width: 600px;
|
||||||
|
outline: none;
|
||||||
|
transition: box-shadow 0.2s, border-color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group input:focus {
|
||||||
|
border-color: var(--theme-color, #f40);
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group button {
|
||||||
|
padding: 9px 15px;
|
||||||
|
background: #1976d2;
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
border-radius: 0 4px 4px 0;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group button:hover {
|
||||||
|
background: #1565c0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
.user-info {
|
||||||
|
background: white;
|
||||||
|
padding: 20px;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
.user-profile {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-details h3 {
|
||||||
|
margin: 0 0 5px 0;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-details p {
|
||||||
|
margin: 0;
|
||||||
|
color: #666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-did-section {
|
||||||
|
margin: 20px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.did-display {
|
||||||
|
padding: 10px;
|
||||||
|
background: #f5f5f5;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 14px;
|
||||||
|
color: #666;
|
||||||
|
word-break: break-all;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.handle-display {
|
||||||
|
padding: 8px 10px;
|
||||||
|
background: #f0f9f0;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 13px;
|
||||||
|
color: #555;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.handle-display strong {
|
||||||
|
color: #2e7d32;
|
||||||
|
}
|
||||||
|
|
||||||
|
.handle-display span {
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 12px;
|
||||||
|
color: #666;
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.pds-display {
|
||||||
|
padding: 8px 10px;
|
||||||
|
background: #e8f4f8;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 13px;
|
||||||
|
color: #555;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-display strong {
|
||||||
|
color: #1976d2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-display span {
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 12px;
|
||||||
|
color: #666;
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collections-section,
|
||||||
|
.records-section {
|
||||||
|
margin: 20px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collections-section h3,
|
||||||
|
.records-section h3 {
|
||||||
|
font-size: 1.2em;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
color: #333;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collections-list,
|
||||||
|
.records-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-link {
|
||||||
|
display: block;
|
||||||
|
padding: 8px 12px;
|
||||||
|
background: #f9f9f9;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 1px solid #e0e0e0;
|
||||||
|
color: #1976d2;
|
||||||
|
text-decoration: none;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 14px;
|
||||||
|
word-break: break-all;
|
||||||
|
transition: all 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-link:hover {
|
||||||
|
background: #e8f4f8;
|
||||||
|
border-color: #1976d2;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-info {
|
||||||
|
padding: 8px 12px;
|
||||||
|
background: #f0f9ff;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 1px solid #b3e5fc;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
color: #1976d2;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collection-info {
|
||||||
|
padding: 8px 12px;
|
||||||
|
background: #f0f9f0;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 1px solid #b3e5b3;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
color: #2e7d32;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collections-header {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collections-toggle {
|
||||||
|
background: #f5f5f5;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 14px;
|
||||||
|
color: #333;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collections-toggle:hover {
|
||||||
|
background: #e8f4f8;
|
||||||
|
border-color: #1976d2;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.pds-test-section,
|
||||||
|
.pds-about-section {
|
||||||
|
margin-bottom: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-test-section h2,
|
||||||
|
.pds-about-section h2 {
|
||||||
|
font-size: 1.8em;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
color: #333;
|
||||||
|
border-bottom: 2px solid #1976d2;
|
||||||
|
padding-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.test-uris {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri {
|
||||||
|
background: #f5f5f5;
|
||||||
|
padding: 15px;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 14px;
|
||||||
|
word-break: break-all;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
border: 1px solid #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri:hover {
|
||||||
|
background: #e8f4f8;
|
||||||
|
border-color: #1976d2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-about-section ul {
|
||||||
|
list-style-type: none;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pds-about-section li {
|
||||||
|
padding: 5px 0;
|
||||||
|
color: #666;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* AT URI Modal Styles */
|
||||||
|
.at-uri-modal-overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-modal-content {
|
||||||
|
background-color: white;
|
||||||
|
border-radius: 8px;
|
||||||
|
max-width: 800px;
|
||||||
|
max-height: 600px;
|
||||||
|
width: 90%;
|
||||||
|
height: 80%;
|
||||||
|
overflow: auto;
|
||||||
|
position: relative;
|
||||||
|
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-modal-close {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: 10px;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
font-size: 20px;
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 1001;
|
||||||
|
padding: 5px 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Loading states */
|
||||||
|
.loading {
|
||||||
|
text-align: center;
|
||||||
|
padding: 20px;
|
||||||
|
color: #666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.error {
|
||||||
|
text-align: center;
|
||||||
|
padding: 20px;
|
||||||
|
color: #d32f2f;
|
||||||
|
background: #ffeaea;
|
||||||
|
border-radius: 4px;
|
||||||
|
margin: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Responsive design */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.pds-search-section {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.pds-search-form {
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: stretch;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group {
|
||||||
|
align-items: stretch;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-group input {
|
||||||
|
width: 100%;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@@ -139,7 +139,7 @@ a.view-markdown:any-link {
|
|||||||
grid-area: header;
|
grid-area: header;
|
||||||
background: #ffffff;
|
background: #ffffff;
|
||||||
border-bottom: 1px solid #d1d9e0;
|
border-bottom: 1px solid #d1d9e0;
|
||||||
padding: 16px 24px;
|
padding: 17px 24px;
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 0;
|
top: 0;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
@@ -723,7 +723,7 @@ article.article-content {
|
|||||||
.footer-social a {
|
.footer-social a {
|
||||||
color: var(--dark-gray) !important;
|
color: var(--dark-gray) !important;
|
||||||
text-decoration: none !important;
|
text-decoration: none !important;
|
||||||
font-size: 20px;
|
font-size: 25px;
|
||||||
transition: all 0.2s ease;
|
transition: all 0.2s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -951,9 +951,11 @@ article.article-content {
|
|||||||
max-width: 100% !important;
|
max-width: 100% !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
.form-group {
|
.form-group {
|
||||||
margin-bottom: 15px !important;
|
margin-bottom: 15px !important;
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
.form-input, .form-textarea {
|
.form-input, .form-textarea {
|
||||||
width: 100% !important;
|
width: 100% !important;
|
||||||
@@ -1838,3 +1840,17 @@ article.article-content {
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
button.ask-at-btn {
|
||||||
|
margin: 10px;
|
||||||
|
background: var(--theme-color);
|
||||||
|
padding: 8px 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
button.ask-at-btn a {
|
||||||
|
color: var(--ai-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
button#searchButton.pds-btn {
|
||||||
|
background: var(--theme-color);
|
||||||
|
}
|
||||||
|
BIN
my-blog/static/img/ue_gasp_dragonik_shin_v0001.png
Normal file
BIN
my-blog/static/img/ue_gasp_dragonik_shin_v0001.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 723 KiB |
370
my-blog/static/js/pds.js
Normal file
370
my-blog/static/js/pds.js
Normal file
@@ -0,0 +1,370 @@
|
|||||||
|
// AT Protocol API functions
|
||||||
|
const AT_PROTOCOL_CONFIG = {
|
||||||
|
primary: {
|
||||||
|
pds: 'https://syu.is',
|
||||||
|
plc: 'https://plc.syu.is',
|
||||||
|
bsky: 'https://bsky.syu.is',
|
||||||
|
web: 'https://web.syu.is'
|
||||||
|
},
|
||||||
|
fallback: {
|
||||||
|
pds: 'https://bsky.social',
|
||||||
|
plc: 'https://plc.directory',
|
||||||
|
bsky: 'https://public.api.bsky.app',
|
||||||
|
web: 'https://bsky.app'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Search user function
|
||||||
|
async function searchUser() {
|
||||||
|
const handleInput = document.getElementById('handleInput');
|
||||||
|
const userInfo = document.getElementById('userInfo');
|
||||||
|
const collectionsList = document.getElementById('collectionsList');
|
||||||
|
const recordsList = document.getElementById('recordsList');
|
||||||
|
const searchButton = document.getElementById('searchButton');
|
||||||
|
|
||||||
|
const input = handleInput.value.trim();
|
||||||
|
if (!input) {
|
||||||
|
alert('Handle nameまたはAT URIを入力してください');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
searchButton.disabled = true;
|
||||||
|
searchButton.innerHTML = '@';
|
||||||
|
//searchButton.innerHTML = '<i class="fab fa-bluesky"></i>';
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Clear previous results
|
||||||
|
document.getElementById('userDidSection').style.display = 'none';
|
||||||
|
document.getElementById('collectionsSection').style.display = 'none';
|
||||||
|
document.getElementById('recordsSection').style.display = 'none';
|
||||||
|
collectionsList.innerHTML = '';
|
||||||
|
recordsList.innerHTML = '';
|
||||||
|
|
||||||
|
// Check if input is AT URI
|
||||||
|
if (input.startsWith('at://')) {
|
||||||
|
// Parse AT URI to check if it's a full record or just a handle/collection
|
||||||
|
const uriParts = input.replace('at://', '').split('/').filter(part => part.length > 0);
|
||||||
|
|
||||||
|
if (uriParts.length >= 3) {
|
||||||
|
// Full AT URI with rkey - show in modal
|
||||||
|
showAtUriModal(input);
|
||||||
|
return;
|
||||||
|
} else if (uriParts.length === 1) {
|
||||||
|
// Just handle in AT URI format (at://handle) - treat as regular handle
|
||||||
|
const handle = uriParts[0];
|
||||||
|
const userProfile = await resolveUserProfile(handle);
|
||||||
|
|
||||||
|
if (userProfile.success) {
|
||||||
|
displayUserDid(userProfile.data);
|
||||||
|
await loadUserCollections(handle, userProfile.data.did);
|
||||||
|
} else {
|
||||||
|
alert('ユーザーが見つかりません: ' + userProfile.error);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} else if (uriParts.length === 2) {
|
||||||
|
// Collection level AT URI - load collection records
|
||||||
|
const [repo, collection] = uriParts;
|
||||||
|
|
||||||
|
try {
|
||||||
|
// First resolve the repo to get handle if it's a DID
|
||||||
|
let handle = repo;
|
||||||
|
if (repo.startsWith('did:')) {
|
||||||
|
// Try to resolve DID to handle - for now just use the DID
|
||||||
|
handle = repo;
|
||||||
|
}
|
||||||
|
|
||||||
|
loadCollectionRecords(handle, collection, repo);
|
||||||
|
} catch (error) {
|
||||||
|
alert('コレクションの読み込みに失敗しました: ' + error.message);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle regular handle search
|
||||||
|
const userProfile = await resolveUserProfile(input);
|
||||||
|
|
||||||
|
if (userProfile.success) {
|
||||||
|
displayUserDid(userProfile.data);
|
||||||
|
await loadUserCollections(input, userProfile.data.did);
|
||||||
|
} else {
|
||||||
|
alert('ユーザーが見つかりません: ' + userProfile.error);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
alert('エラーが発生しました: ' + error.message);
|
||||||
|
} finally {
|
||||||
|
searchButton.disabled = false;
|
||||||
|
searchButton.innerHTML = '@';
|
||||||
|
//searchButton.innerHTML = '<i class="fab fa-bluesky"></i>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resolve user profile
|
||||||
|
async function resolveUserProfile(handle) {
|
||||||
|
try {
|
||||||
|
let response = null;
|
||||||
|
|
||||||
|
// Try syu.is first
|
||||||
|
try {
|
||||||
|
response = await fetch(`${AT_PROTOCOL_CONFIG.primary.pds}/xrpc/com.atproto.repo.describeRepo?repo=${handle}`);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('Failed to resolve from syu.is:', error);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If syu.is fails, try bsky.social
|
||||||
|
if (!response || !response.ok) {
|
||||||
|
response = await fetch(`${AT_PROTOCOL_CONFIG.fallback.pds}/xrpc/com.atproto.repo.describeRepo?repo=${handle}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to resolve handle');
|
||||||
|
}
|
||||||
|
|
||||||
|
const repoData = await response.json();
|
||||||
|
|
||||||
|
// Get profile data
|
||||||
|
const profileResponse = await fetch(`${AT_PROTOCOL_CONFIG.fallback.bsky}/xrpc/app.bsky.actor.getProfile?actor=${repoData.did}`);
|
||||||
|
const profileData = await profileResponse.json();
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
data: {
|
||||||
|
did: repoData.did,
|
||||||
|
handle: profileData.handle,
|
||||||
|
displayName: profileData.displayName,
|
||||||
|
avatar: profileData.avatar,
|
||||||
|
description: profileData.description,
|
||||||
|
pds: repoData.didDoc.service.find(s => s.type === 'AtprotoPersonalDataServer')?.serviceEndpoint
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: error.message
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display user DID
|
||||||
|
function displayUserDid(profile) {
|
||||||
|
document.getElementById('userPdsText').textContent = profile.pds || 'Unknown';
|
||||||
|
document.getElementById('userHandleText').textContent = profile.handle;
|
||||||
|
document.getElementById('userDidText').textContent = profile.did;
|
||||||
|
document.getElementById('userDidSection').style.display = 'block';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load user collections
|
||||||
|
async function loadUserCollections(handle, did) {
|
||||||
|
const collectionsList = document.getElementById('collectionsList');
|
||||||
|
|
||||||
|
collectionsList.innerHTML = '<div class="loading">コレクションを読み込み中...</div>';
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Try to get collections from describeRepo
|
||||||
|
let response = await fetch(`${AT_PROTOCOL_CONFIG.primary.pds}/xrpc/com.atproto.repo.describeRepo?repo=${handle}`);
|
||||||
|
let usedPds = AT_PROTOCOL_CONFIG.primary.pds;
|
||||||
|
|
||||||
|
// If syu.is fails, try bsky.social
|
||||||
|
if (!response.ok) {
|
||||||
|
response = await fetch(`${AT_PROTOCOL_CONFIG.fallback.pds}/xrpc/com.atproto.repo.describeRepo?repo=${handle}`);
|
||||||
|
usedPds = AT_PROTOCOL_CONFIG.fallback.pds;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to describe repository');
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
const collections = data.collections || [];
|
||||||
|
|
||||||
|
// Display collections as AT URI links
|
||||||
|
collectionsList.innerHTML = '';
|
||||||
|
if (collections.length === 0) {
|
||||||
|
collectionsList.innerHTML = '<div class="error">コレクションが見つかりませんでした</div>';
|
||||||
|
} else {
|
||||||
|
|
||||||
|
collections.forEach(collection => {
|
||||||
|
const atUri = `at://${did}/${collection}/`;
|
||||||
|
const collectionElement = document.createElement('a');
|
||||||
|
collectionElement.className = 'at-uri-link';
|
||||||
|
collectionElement.href = '#';
|
||||||
|
collectionElement.textContent = atUri;
|
||||||
|
collectionElement.onclick = (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
loadCollectionRecords(handle, collection, did);
|
||||||
|
// Close collections and update toggle
|
||||||
|
document.getElementById('collectionsList').style.display = 'none';
|
||||||
|
document.getElementById('collectionsToggle').textContent = '[-] Collections';
|
||||||
|
};
|
||||||
|
collectionsList.appendChild(collectionElement);
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('collectionsSection').style.display = 'block';
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
collectionsList.innerHTML = '<div class="error">コレクションの読み込みに失敗しました: ' + error.message + '</div>';
|
||||||
|
document.getElementById('collectionsSection').style.display = 'block';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load collection records
|
||||||
|
async function loadCollectionRecords(handle, collection, did) {
|
||||||
|
const recordsList = document.getElementById('recordsList');
|
||||||
|
|
||||||
|
recordsList.innerHTML = '<div class="loading">レコードを読み込み中...</div>';
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Try with syu.is first
|
||||||
|
let response = await fetch(`${AT_PROTOCOL_CONFIG.primary.pds}/xrpc/com.atproto.repo.listRecords?repo=${handle}&collection=${collection}`);
|
||||||
|
let usedPds = AT_PROTOCOL_CONFIG.primary.pds;
|
||||||
|
|
||||||
|
// If that fails, try with bsky.social
|
||||||
|
if (!response.ok) {
|
||||||
|
response = await fetch(`${AT_PROTOCOL_CONFIG.fallback.pds}/xrpc/com.atproto.repo.listRecords?repo=${handle}&collection=${collection}`);
|
||||||
|
usedPds = AT_PROTOCOL_CONFIG.fallback.pds;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to load records');
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
|
||||||
|
// Display records as AT URI links
|
||||||
|
recordsList.innerHTML = '';
|
||||||
|
|
||||||
|
// Add collection info for records
|
||||||
|
const collectionInfo = document.createElement('div');
|
||||||
|
collectionInfo.className = 'collection-info';
|
||||||
|
collectionInfo.innerHTML = `<strong>${collection}</strong>`;
|
||||||
|
recordsList.appendChild(collectionInfo);
|
||||||
|
|
||||||
|
data.records.forEach(record => {
|
||||||
|
const atUri = record.uri;
|
||||||
|
const recordElement = document.createElement('a');
|
||||||
|
recordElement.className = 'at-uri-link';
|
||||||
|
recordElement.href = '#';
|
||||||
|
recordElement.textContent = atUri;
|
||||||
|
recordElement.onclick = (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
showAtUriModal(atUri);
|
||||||
|
};
|
||||||
|
recordsList.appendChild(recordElement);
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('recordsSection').style.display = 'block';
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
recordsList.innerHTML = '<div class="error">レコードの読み込みに失敗しました: ' + error.message + '</div>';
|
||||||
|
document.getElementById('recordsSection').style.display = 'block';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show AT URI modal
|
||||||
|
function showAtUriModal(uri) {
|
||||||
|
const modal = document.getElementById('atUriModal');
|
||||||
|
const content = document.getElementById('atUriContent');
|
||||||
|
|
||||||
|
content.innerHTML = '<div class="loading">レコードを読み込み中...</div>';
|
||||||
|
modal.style.display = 'flex';
|
||||||
|
|
||||||
|
// Load record data
|
||||||
|
loadAtUriRecord(uri, content);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load AT URI record
|
||||||
|
async function loadAtUriRecord(uri, contentElement) {
|
||||||
|
try {
|
||||||
|
const parts = uri.replace('at://', '').split('/');
|
||||||
|
const repo = parts[0];
|
||||||
|
const collection = parts[1];
|
||||||
|
const rkey = parts[2];
|
||||||
|
|
||||||
|
// Try with syu.is first
|
||||||
|
let response = await fetch(`${AT_PROTOCOL_CONFIG.primary.pds}/xrpc/com.atproto.repo.getRecord?repo=${repo}&collection=${collection}&rkey=${rkey}`);
|
||||||
|
|
||||||
|
// If that fails, try with bsky.social
|
||||||
|
if (!response.ok) {
|
||||||
|
response = await fetch(`${AT_PROTOCOL_CONFIG.fallback.pds}/xrpc/com.atproto.repo.getRecord?repo=${repo}&collection=${collection}&rkey=${rkey}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to load record');
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await response.json();
|
||||||
|
|
||||||
|
contentElement.innerHTML = `
|
||||||
|
<div style="padding: 20px;">
|
||||||
|
<h3>AT URI Record</h3>
|
||||||
|
<div style="font-family: monospace; font-size: 14px; color: #666; margin-bottom: 20px; word-break: break-all;">
|
||||||
|
${uri}
|
||||||
|
</div>
|
||||||
|
<div style="font-size: 12px; color: #999; margin-bottom: 20px;">
|
||||||
|
Repo: ${repo} | Collection: ${collection} | RKey: ${rkey}
|
||||||
|
</div>
|
||||||
|
<h4>Record Data</h4>
|
||||||
|
<pre style="background: #f5f5f5; padding: 15px; border-radius: 4px; overflow: auto;">${JSON.stringify(data, null, 2)}</pre>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
} catch (error) {
|
||||||
|
contentElement.innerHTML = `
|
||||||
|
<div style="padding: 20px; color: red;">
|
||||||
|
<strong>Error:</strong> ${error.message}
|
||||||
|
<div style="margin-top: 10px; font-size: 12px;">
|
||||||
|
<strong>URI:</strong> ${uri}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close AT URI modal
|
||||||
|
function closeAtUriModal(event) {
|
||||||
|
const modal = document.getElementById('atUriModal');
|
||||||
|
if (event && event.target !== modal) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
modal.style.display = 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize AT URI click handlers
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
// Add click handlers to existing AT URIs
|
||||||
|
document.querySelectorAll('.at-uri').forEach(element => {
|
||||||
|
element.addEventListener('click', function() {
|
||||||
|
const uri = this.getAttribute('data-at-uri');
|
||||||
|
showAtUriModal(uri);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// ESC key to close modal
|
||||||
|
document.addEventListener('keydown', function(event) {
|
||||||
|
if (event.key === 'Escape') {
|
||||||
|
closeAtUriModal();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Enter key to search
|
||||||
|
document.getElementById('handleInput').addEventListener('keydown', function(event) {
|
||||||
|
if (event.key === 'Enter') {
|
||||||
|
searchUser();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
// Toggle collections visibility
|
||||||
|
function toggleCollections() {
|
||||||
|
const collectionsList = document.getElementById('collectionsList');
|
||||||
|
const toggleButton = document.getElementById('collectionsToggle');
|
||||||
|
|
||||||
|
if (collectionsList.style.display === 'none') {
|
||||||
|
collectionsList.style.display = 'block';
|
||||||
|
toggleButton.textContent = '[-] Collections';
|
||||||
|
} else {
|
||||||
|
collectionsList.style.display = 'none';
|
||||||
|
toggleButton.textContent = '[+] Collections';
|
||||||
|
}
|
||||||
|
}
|
61
my-blog/templates/at-browser-assets.html
Normal file
61
my-blog/templates/at-browser-assets.html
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<!-- AT Browser Integration - Temporarily disabled to fix site display -->
|
||||||
|
<!--
|
||||||
|
<script crossorigin src="https://unpkg.com/react@18/umd/react.production.min.js"></script>
|
||||||
|
<script crossorigin src="https://unpkg.com/react-dom@18/umd/react-dom.production.min.js"></script>
|
||||||
|
<script src="/assets/pds-browser.umd.js"></script>
|
||||||
|
<script>
|
||||||
|
// AT Browser integration - needs debugging
|
||||||
|
console.log('AT Browser integration temporarily disabled');
|
||||||
|
</script>
|
||||||
|
-->
|
||||||
|
|
||||||
|
<style>
|
||||||
|
/* AT Browser Modal Styles */
|
||||||
|
.at-uri-modal-overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-modal-content {
|
||||||
|
background-color: white;
|
||||||
|
border-radius: 8px;
|
||||||
|
max-width: 800px;
|
||||||
|
max-height: 600px;
|
||||||
|
width: 90%;
|
||||||
|
height: 80%;
|
||||||
|
overflow: auto;
|
||||||
|
position: relative;
|
||||||
|
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-modal-close {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: 10px;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
font-size: 20px;
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 1001;
|
||||||
|
padding: 5px 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* AT URI Link Styles */
|
||||||
|
[data-at-uri] {
|
||||||
|
color: #1976d2;
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-at-uri]:hover {
|
||||||
|
color: #1565c0;
|
||||||
|
}
|
||||||
|
</style>
|
@@ -12,6 +12,7 @@
|
|||||||
<!-- Stylesheets -->
|
<!-- Stylesheets -->
|
||||||
<link rel="stylesheet" href="/css/style.css">
|
<link rel="stylesheet" href="/css/style.css">
|
||||||
<link rel="stylesheet" href="/css/svg-animation-package.css">
|
<link rel="stylesheet" href="/css/svg-animation-package.css">
|
||||||
|
<link rel="stylesheet" href="/css/pds.css">
|
||||||
<link rel="stylesheet" href="/pkg/icomoon/style.css">
|
<link rel="stylesheet" href="/pkg/icomoon/style.css">
|
||||||
<link rel="stylesheet" href="/pkg/font-awesome/css/all.min.css">
|
<link rel="stylesheet" href="/pkg/font-awesome/css/all.min.css">
|
||||||
|
|
||||||
@@ -48,7 +49,18 @@
|
|||||||
</svg>
|
</svg>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="header-actions">
|
<div class="header-actions">
|
||||||
|
<!-- User Handle Input Form -->
|
||||||
|
<div class="pds-search-section">
|
||||||
|
<form class="pds-search-form" onsubmit="searchUser(); return false;">
|
||||||
|
<div class="form-group">
|
||||||
|
<input type="text" id="handleInput" placeholder="at://syui.ai" value="syui.ai" />
|
||||||
|
<button type="submit" id="searchButton" class="pds-btn">
|
||||||
|
@
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
<button class="ask-ai-btn" onclick="toggleAskAI()" id="askAiButton">
|
<button class="ask-ai-btn" onclick="toggleAskAI()" id="askAiButton">
|
||||||
<span class="ai-icon icon-ai"></span>
|
<span class="ai-icon icon-ai"></span>
|
||||||
ai
|
ai
|
||||||
@@ -77,6 +89,9 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
|
<!-- Pds Panel -->
|
||||||
|
{% include "pds-header.html" %}
|
||||||
|
|
||||||
{% block content %}{% endblock %}
|
{% block content %}{% endblock %}
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
@@ -113,6 +128,7 @@
|
|||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
<script src="/js/ask-ai.js"></script>
|
<script src="/js/ask-ai.js"></script>
|
||||||
|
<script src="/js/pds.js"></script>
|
||||||
<script src="/js/theme.js"></script>
|
<script src="/js/theme.js"></script>
|
||||||
<script src="/js/image-comparison.js"></script>
|
<script src="/js/image-comparison.js"></script>
|
||||||
|
|
||||||
@@ -131,5 +147,6 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
{% include "oauth-assets.html" %}
|
{% include "oauth-assets.html" %}
|
||||||
|
{% include "at-browser-assets.html" %}
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
135
my-blog/templates/game.html
Normal file
135
my-blog/templates/game.html
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
{% extends "base.html" %}
|
||||||
|
|
||||||
|
{% block title %}Game - {{ config.title }}{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div id="gameContainer" class="game-container">
|
||||||
|
<div id="gameAuth" class="game-auth-section">
|
||||||
|
<h1>Login to Play</h1>
|
||||||
|
<p>Please authenticate with your AT Protocol account to access the game.</p>
|
||||||
|
<div id="authRoot"></div>
|
||||||
|
</div>
|
||||||
|
<div id="gameFrame" class="game-frame-container" style="display: none;">
|
||||||
|
<iframe
|
||||||
|
id="pixelStreamingFrame"
|
||||||
|
src="https://verse.syui.ai/simple-noui.html"
|
||||||
|
frameborder="0"
|
||||||
|
allowfullscreen
|
||||||
|
allow="microphone; camera; fullscreen; autoplay"
|
||||||
|
class="pixel-streaming-iframe"
|
||||||
|
></iframe>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
/* Game specific styles */
|
||||||
|
.game-container {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
background: #000;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-auth-section {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 100vh;
|
||||||
|
background: linear-gradient(135deg, #1a1a1a 0%, #2d2d2d 100%);
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-auth-section h1 {
|
||||||
|
font-size: 2.5em;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-auth-section p {
|
||||||
|
font-size: 1.2em;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-frame-container {
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pixel-streaming-iframe {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Override auth button for game page */
|
||||||
|
.game-auth-section .auth-section {
|
||||||
|
background: transparent;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-auth-section .auth-button {
|
||||||
|
font-size: 1.2em;
|
||||||
|
padding: 12px 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide header and footer on game page */
|
||||||
|
body:has(.game-container) header,
|
||||||
|
body:has(.game-container) footer,
|
||||||
|
body:has(.game-container) nav {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove any body padding/margin for full screen game */
|
||||||
|
body:has(.game-container) {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Wait for OAuth component to be loaded
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
// Check if user is already authenticated
|
||||||
|
const checkAuthStatus = () => {
|
||||||
|
// Check if OAuth components are available and user is authenticated
|
||||||
|
if (window.currentUser && window.currentAgent) {
|
||||||
|
showGame();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Show game iframe
|
||||||
|
const showGame = () => {
|
||||||
|
document.getElementById('gameAuth').style.display = 'none';
|
||||||
|
document.getElementById('gameFrame').style.display = 'block';
|
||||||
|
};
|
||||||
|
|
||||||
|
// Listen for OAuth success
|
||||||
|
window.addEventListener('oauth-success', function(event) {
|
||||||
|
console.log('OAuth success:', event.detail);
|
||||||
|
showGame();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check auth status on load
|
||||||
|
if (!checkAuthStatus()) {
|
||||||
|
// Check periodically if OAuth components are loaded
|
||||||
|
const authCheckInterval = setInterval(() => {
|
||||||
|
if (checkAuthStatus()) {
|
||||||
|
clearInterval(authCheckInterval);
|
||||||
|
}
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- Include OAuth assets -->
|
||||||
|
{% include "oauth-assets.html" %}
|
||||||
|
{% endblock %}
|
48
my-blog/templates/pds-header.html
Normal file
48
my-blog/templates/pds-header.html
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
<div class="pds-container">
|
||||||
|
<div class="pds-header">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<!-- Current User DID -->
|
||||||
|
<div id="userDidSection" class="user-did-section" style="display: none;">
|
||||||
|
<div class="pds-display">
|
||||||
|
<strong>PDS:</strong> <span id="userPdsText"></span>
|
||||||
|
</div>
|
||||||
|
<div class="handle-display">
|
||||||
|
<strong>Handle:</strong> <span id="userHandleText"></span>
|
||||||
|
</div>
|
||||||
|
<div class="did-display">
|
||||||
|
<span id="userDidText"></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Collection List -->
|
||||||
|
<div id="collectionsSection" class="collections-section" style="display: none;">
|
||||||
|
<div class="collections-header">
|
||||||
|
<button id="collectionsToggle" class="collections-toggle" onclick="toggleCollections()">[+] Collections</button>
|
||||||
|
</div>
|
||||||
|
<div id="collectionsList" class="collections-list" style="display: none;">
|
||||||
|
<!-- Collections will be populated here -->
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- AT URI Records -->
|
||||||
|
<div id="recordsSection" class="records-section" style="display: none;">
|
||||||
|
<div id="recordsList" class="records-list">
|
||||||
|
<!-- Records will be populated here -->
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- AT URI Modal -->
|
||||||
|
<div id="atUriModal" class="at-uri-modal-overlay" style="display: none;" onclick="closeAtUriModal(event)">
|
||||||
|
<div class="at-uri-modal-content">
|
||||||
|
<button class="at-uri-modal-close" onclick="closeAtUriModal()">×</button>
|
||||||
|
<div id="atUriContent"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
6
my-blog/templates/pds.html
Normal file
6
my-blog/templates/pds.html
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{% extends "base.html" %}
|
||||||
|
|
||||||
|
{% block title %}at-uri browser - {{ config.title }}{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
{% endblock %}
|
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "ailog-oauth",
|
"name": "ailog-oauth",
|
||||||
"version": "0.2.9",
|
"version": "0.3.1",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
|
@@ -126,11 +126,11 @@ body {
|
|||||||
|
|
||||||
/* Header */
|
/* Header */
|
||||||
.oauth-app-header {
|
.oauth-app-header {
|
||||||
background: var(--background);
|
background: var(--background);
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 0;
|
top: 0;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.oauth-header-content {
|
.oauth-header-content {
|
||||||
@@ -139,7 +139,7 @@ body {
|
|||||||
/* align-items: center; */
|
/* align-items: center; */
|
||||||
max-width: 800px;
|
max-width: 800px;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
padding: 25px 0;
|
padding: 30px 0;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -287,7 +287,6 @@ body {
|
|||||||
.auth-section {
|
.auth-section {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 8px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.auth-section.search-bar-layout {
|
.auth-section.search-bar-layout {
|
||||||
@@ -302,10 +301,10 @@ body {
|
|||||||
.auth-section.search-bar-layout .handle-input {
|
.auth-section.search-bar-layout .handle-input {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 10px 15px;
|
padding: 9px 15px;
|
||||||
font-size: 16px;
|
font-size: 13px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
border-radius: 8px 0 0 8px;
|
border-radius: 4px 0 0 4px;
|
||||||
background: var(--background);
|
background: var(--background);
|
||||||
outline: none;
|
outline: none;
|
||||||
transition: border-color 0.2s;
|
transition: border-color 0.2s;
|
||||||
@@ -319,12 +318,13 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.auth-section.search-bar-layout .auth-button {
|
.auth-section.search-bar-layout .auth-button {
|
||||||
border-radius: 0 6px 6px 0;
|
border-radius: 0 4px 4px 0;
|
||||||
border: 1px solid var(--primary);
|
border: 1px solid var(--primary);
|
||||||
border-left: none;
|
border-left: none;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 10px 15px;
|
padding: 9px 15px;
|
||||||
height: 40px;
|
min-width: 50px;
|
||||||
|
min-height: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Auth Button */
|
/* Auth Button */
|
||||||
@@ -332,15 +332,26 @@ body {
|
|||||||
background: var(--primary);
|
background: var(--primary);
|
||||||
color: white;
|
color: white;
|
||||||
border: none;
|
border: none;
|
||||||
border-radius: 6px;
|
border-radius: 4px;
|
||||||
padding: 8px 16px;
|
padding: 9px 15px;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: background 0.2s;
|
transition: background 0.2s;
|
||||||
height: 40px;
|
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
min-width: 50px;
|
||||||
|
min-height: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Loading spinner for auth button */
|
||||||
|
.auth-button.loading i {
|
||||||
|
animation: spin 1s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
from { transform: rotate(0deg); }
|
||||||
|
to { transform: rotate(360deg); }
|
||||||
}
|
}
|
||||||
|
|
||||||
.auth-button:hover {
|
.auth-button:hover {
|
||||||
@@ -422,10 +433,6 @@ body {
|
|||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.form-group {
|
|
||||||
margin-bottom: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.form-group label {
|
.form-group label {
|
||||||
display: block;
|
display: block;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
@@ -919,10 +926,6 @@ body {
|
|||||||
max-width: 100% !important;
|
max-width: 100% !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.form-group {
|
|
||||||
margin-bottom: 15px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.form-input, .form-textarea {
|
.form-input, .form-textarea {
|
||||||
width: 100% !important;
|
width: 100% !important;
|
||||||
max-width: 100% !important;
|
max-width: 100% !important;
|
||||||
|
@@ -118,6 +118,14 @@ export default function App() {
|
|||||||
}
|
}
|
||||||
}, [adminData])
|
}, [adminData])
|
||||||
|
|
||||||
|
// Expose current user and agent for game page
|
||||||
|
useEffect(() => {
|
||||||
|
if (user && agent) {
|
||||||
|
window.currentUser = user
|
||||||
|
window.currentAgent = agent
|
||||||
|
}
|
||||||
|
}, [user, agent])
|
||||||
|
|
||||||
// Event listeners for blog communication
|
// Event listeners for blog communication
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// Clear OAuth completion flag once app is loaded
|
// Clear OAuth completion flag once app is loaded
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
import React, { useState } from 'react'
|
import React, { useState } from 'react'
|
||||||
|
import { logger } from '../utils/logger.js'
|
||||||
|
|
||||||
export default function AuthButton({ user, onLogin, onLogout, loading }) {
|
export default function AuthButton({ user, onLogin, onLogout, loading }) {
|
||||||
const [handleInput, setHandleInput] = useState('')
|
const [handleInput, setHandleInput] = useState('')
|
||||||
@@ -12,7 +13,7 @@ export default function AuthButton({ user, onLogin, onLogout, loading }) {
|
|||||||
try {
|
try {
|
||||||
await onLogin(handleInput.trim())
|
await onLogin(handleInput.trim())
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Login failed:', error)
|
logger.error('Login failed:', error)
|
||||||
alert('ログインに失敗しました: ' + error.message)
|
alert('ログインに失敗しました: ' + error.message)
|
||||||
} finally {
|
} finally {
|
||||||
setIsLoading(false)
|
setIsLoading(false)
|
||||||
@@ -68,9 +69,9 @@ export default function AuthButton({ user, onLogin, onLogout, loading }) {
|
|||||||
type="button"
|
type="button"
|
||||||
onClick={handleSubmit}
|
onClick={handleSubmit}
|
||||||
disabled={isLoading || !handleInput.trim()}
|
disabled={isLoading || !handleInput.trim()}
|
||||||
className="auth-button"
|
className={`auth-button ${isLoading ? 'loading' : ''}`}
|
||||||
>
|
>
|
||||||
{isLoading ? 'Loading...' : <i className="fab fa-bluesky"></i>}
|
<i className={isLoading ? "fas fa-spinner" : "fab fa-bluesky"}></i>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import React, { useState, useEffect } from 'react'
|
import React, { useState, useEffect } from 'react'
|
||||||
import Avatar, { AvatarWithCard, AvatarList } from './Avatar.jsx'
|
import Avatar, { AvatarWithCard, AvatarList } from './Avatar.jsx'
|
||||||
import { getAvatar, batchFetchAvatars, prefetchAvatar } from '../utils/avatar.js'
|
import { getAvatar, batchFetchAvatars, prefetchAvatar } from '../utils/avatar.js'
|
||||||
|
import { logger } from '../utils/logger.js'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test component to demonstrate avatar functionality
|
* Test component to demonstrate avatar functionality
|
||||||
@@ -63,7 +64,7 @@ export default function AvatarTest() {
|
|||||||
|
|
||||||
setTestResults(results)
|
setTestResults(results)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Test failed:', error)
|
logger.error('Test failed:', error)
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -78,7 +79,7 @@ export default function AvatarTest() {
|
|||||||
batchResults: Object.fromEntries(avatarMap)
|
batchResults: Object.fromEntries(avatarMap)
|
||||||
}))
|
}))
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Batch test failed:', error)
|
logger.error('Batch test failed:', error)
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -94,7 +95,7 @@ export default function AvatarTest() {
|
|||||||
prefetchResult: cachedAvatar
|
prefetchResult: cachedAvatar
|
||||||
}))
|
}))
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Prefetch test failed:', error)
|
logger.error('Prefetch test failed:', error)
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import React, { useState } from 'react'
|
import React, { useState } from 'react'
|
||||||
import { atproto, collections } from '../api/atproto.js'
|
import { atproto, collections } from '../api/atproto.js'
|
||||||
import { env } from '../config/env.js'
|
import { env } from '../config/env.js'
|
||||||
|
import { logger } from '../utils/logger.js'
|
||||||
|
|
||||||
const ProfileForm = ({ user, agent, apiConfig, onProfilePosted }) => {
|
const ProfileForm = ({ user, agent, apiConfig, onProfilePosted }) => {
|
||||||
const [text, setText] = useState('')
|
const [text, setText] = useState('')
|
||||||
@@ -79,7 +80,7 @@ const ProfileForm = ({ user, agent, apiConfig, onProfilePosted }) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('Failed to create profile:', err)
|
logger.error('Failed to create profile:', err)
|
||||||
setError(err.message || 'プロフィールの作成に失敗しました')
|
setError(err.message || 'プロフィールの作成に失敗しました')
|
||||||
} finally {
|
} finally {
|
||||||
setPosting(false)
|
setPosting(false)
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import React, { useState } from 'react'
|
import React, { useState } from 'react'
|
||||||
import { atproto } from '../api/atproto.js'
|
import { atproto } from '../api/atproto.js'
|
||||||
import { getPdsFromHandle, getApiConfig } from '../utils/pds.js'
|
import { getPdsFromHandle, getApiConfig } from '../utils/pds.js'
|
||||||
|
import { logger } from '../utils/logger.js'
|
||||||
|
|
||||||
export default function UserLookup() {
|
export default function UserLookup() {
|
||||||
const [handleInput, setHandleInput] = useState('')
|
const [handleInput, setHandleInput] = useState('')
|
||||||
@@ -26,7 +27,7 @@ export default function UserLookup() {
|
|||||||
config: apiConfig
|
config: apiConfig
|
||||||
})
|
})
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('User lookup failed:', error)
|
logger.error('User lookup failed:', error)
|
||||||
setUserInfo({ error: error.message })
|
setUserInfo({ error: error.message })
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
import { useState, useEffect } from 'react'
|
import { useState, useEffect } from 'react'
|
||||||
import { OAuthService } from '../services/oauth.js'
|
import { OAuthService } from '../services/oauth.js'
|
||||||
|
import { logger } from '../utils/logger.js'
|
||||||
|
|
||||||
const oauthService = new OAuthService()
|
const oauthService = new OAuthService()
|
||||||
|
|
||||||
@@ -21,7 +22,7 @@ export function useAuth() {
|
|||||||
|
|
||||||
// If we're on callback page and authentication succeeded, notify parent
|
// If we're on callback page and authentication succeeded, notify parent
|
||||||
if (window.location.pathname === '/oauth/callback') {
|
if (window.location.pathname === '/oauth/callback') {
|
||||||
console.log('OAuth callback completed, notifying parent window')
|
logger.log('OAuth callback completed, notifying parent window')
|
||||||
|
|
||||||
// Get referrer or use stored return URL
|
// Get referrer or use stored return URL
|
||||||
const returnUrl = sessionStorage.getItem('oauth_return_url') ||
|
const returnUrl = sessionStorage.getItem('oauth_return_url') ||
|
||||||
@@ -48,7 +49,7 @@ export function useAuth() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Auth initialization failed:', error)
|
logger.error('Auth initialization failed:', error)
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
|
@@ -2,6 +2,7 @@ import { BrowserOAuthClient } from '@atproto/oauth-client-browser'
|
|||||||
import { Agent } from '@atproto/api'
|
import { Agent } from '@atproto/api'
|
||||||
import { env } from '../config/env.js'
|
import { env } from '../config/env.js'
|
||||||
import { isSyuIsHandle } from '../utils/pds.js'
|
import { isSyuIsHandle } from '../utils/pds.js'
|
||||||
|
import { logger } from '../utils/logger.js'
|
||||||
|
|
||||||
export class OAuthService {
|
export class OAuthService {
|
||||||
constructor() {
|
constructor() {
|
||||||
@@ -44,7 +45,7 @@ export class OAuthService {
|
|||||||
// Try to restore session
|
// Try to restore session
|
||||||
return await this.restoreSession()
|
return await this.restoreSession()
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('OAuth initialization failed:', error)
|
logger.error('OAuth initialization failed:', error)
|
||||||
this.initPromise = null
|
this.initPromise = null
|
||||||
throw error
|
throw error
|
||||||
}
|
}
|
||||||
@@ -89,18 +90,18 @@ export class OAuthService {
|
|||||||
displayName = profile.data.displayName || null
|
displayName = profile.data.displayName || null
|
||||||
avatar = profile.data.avatar || null
|
avatar = profile.data.avatar || null
|
||||||
|
|
||||||
console.log('Profile fetched from session:', {
|
logger.log('Profile fetched from session:', {
|
||||||
did,
|
did,
|
||||||
handle,
|
handle,
|
||||||
displayName,
|
displayName,
|
||||||
avatar: avatar ? 'present' : 'none'
|
avatar: avatar ? 'present' : 'none'
|
||||||
})
|
})
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.log('Failed to get profile from session:', error)
|
logger.log('Failed to get profile from session:', error)
|
||||||
// Keep the basic info we have
|
// Keep the basic info we have
|
||||||
}
|
}
|
||||||
} else if (did && did.includes('test-')) {
|
} else if (did && did.includes('test-')) {
|
||||||
console.log('Skipping profile fetch for test DID:', did)
|
logger.log('Skipping profile fetch for test DID:', did)
|
||||||
}
|
}
|
||||||
|
|
||||||
this.sessionInfo = {
|
this.sessionInfo = {
|
||||||
@@ -140,7 +141,7 @@ export class OAuthService {
|
|||||||
}
|
}
|
||||||
return null
|
return null
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Auth check failed:', error)
|
logger.error('Auth check failed:', error)
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -168,7 +169,7 @@ export class OAuthService {
|
|||||||
// Reload page
|
// Reload page
|
||||||
window.location.reload()
|
window.location.reload()
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Logout failed:', error)
|
logger.error('Logout failed:', error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
12
pds/index.html
Normal file
12
pds/index.html
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="ja">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>AT URI Browser - syui.ai</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="root"></div>
|
||||||
|
<script type="module" src="/src/main.jsx"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
27
pds/package.json
Normal file
27
pds/package.json
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"name": "pds-browser",
|
||||||
|
"version": "0.3.1",
|
||||||
|
"description": "AT Protocol browser for ai.log",
|
||||||
|
"main": "index.js",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "vite",
|
||||||
|
"build": "vite build",
|
||||||
|
"preview": "vite preview"
|
||||||
|
},
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@atproto/api": "^0.13.0",
|
||||||
|
"@atproto/did": "^0.1.0",
|
||||||
|
"@atproto/lexicon": "^0.4.0",
|
||||||
|
"@atproto/syntax": "^0.3.0",
|
||||||
|
"react": "^18.2.0",
|
||||||
|
"react-dom": "^18.2.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/react": "^18.0.37",
|
||||||
|
"@types/react-dom": "^18.0.11",
|
||||||
|
"@vitejs/plugin-react": "^4.0.0",
|
||||||
|
"vite": "^5.0.0"
|
||||||
|
}
|
||||||
|
}
|
128
pds/src/App.css
Normal file
128
pds/src/App.css
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
body {
|
||||||
|
font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||||
|
margin: 0;
|
||||||
|
padding: 20px;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
line-height: 1.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
max-width: 1200px;
|
||||||
|
margin: 0 auto;
|
||||||
|
background: white;
|
||||||
|
padding: 30px;
|
||||||
|
border-radius: 10px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
color: #333;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
border-bottom: 3px solid #007acc;
|
||||||
|
padding-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.test-section {
|
||||||
|
margin-bottom: 30px;
|
||||||
|
padding: 20px;
|
||||||
|
background: #f8f9fa;
|
||||||
|
border-radius: 8px;
|
||||||
|
border-left: 4px solid #007acc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.test-uris {
|
||||||
|
background: #fff;
|
||||||
|
padding: 15px;
|
||||||
|
border-radius: 5px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
margin: 15px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri {
|
||||||
|
font-family: 'Monaco', 'Consolas', monospace;
|
||||||
|
background: #f4f4f4;
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 4px;
|
||||||
|
margin: 10px 0;
|
||||||
|
display: block;
|
||||||
|
word-break: break-all;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri:hover {
|
||||||
|
background: #e8e8e8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions {
|
||||||
|
background: #e8f4f8;
|
||||||
|
padding: 15px;
|
||||||
|
border-radius: 5px;
|
||||||
|
margin: 15px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.instructions ol {
|
||||||
|
margin: 10px 0;
|
||||||
|
padding-left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.back-link {
|
||||||
|
display: inline-block;
|
||||||
|
margin-top: 20px;
|
||||||
|
color: #007acc;
|
||||||
|
text-decoration: none;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.back-link:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* AT Browser Modal Styles */
|
||||||
|
.at-uri-modal-overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-modal-content {
|
||||||
|
background-color: white;
|
||||||
|
border-radius: 8px;
|
||||||
|
max-width: 800px;
|
||||||
|
max-height: 600px;
|
||||||
|
width: 90%;
|
||||||
|
height: 80%;
|
||||||
|
overflow: auto;
|
||||||
|
position: relative;
|
||||||
|
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.at-uri-modal-close {
|
||||||
|
position: absolute;
|
||||||
|
top: 10px;
|
||||||
|
right: 10px;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
font-size: 20px;
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 1001;
|
||||||
|
padding: 5px 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* AT URI Link Styles */
|
||||||
|
[data-at-uri] {
|
||||||
|
color: #1976d2;
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-at-uri]:hover {
|
||||||
|
color: #1565c0;
|
||||||
|
}
|
62
pds/src/App.jsx
Normal file
62
pds/src/App.jsx
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import { AtUriBrowser } from './components/AtUriBrowser.jsx'
|
||||||
|
import './App.css'
|
||||||
|
|
||||||
|
function App() {
|
||||||
|
return (
|
||||||
|
<AtUriBrowser>
|
||||||
|
<div className="container">
|
||||||
|
<h1>AT URI Browser</h1>
|
||||||
|
|
||||||
|
<div className="test-section">
|
||||||
|
<h2>テスト用 AT URI</h2>
|
||||||
|
<p>以下のAT URIをクリックすると、モーダルでコンテンツが表示されます。</p>
|
||||||
|
|
||||||
|
<div className="test-uris">
|
||||||
|
<div className="at-uri" data-at-uri="at://did:plc:vzsvtbtbnwn22xjqhcu3vd6y/app.bsky.feed.post/3lu5givmkc222">
|
||||||
|
at://did:plc:vzsvtbtbnwn22xjqhcu3vd6y/app.bsky.feed.post/3lu5givmkc222
|
||||||
|
</div>
|
||||||
|
<div className="at-uri" data-at-uri="at://did:plc:vzsvtbtbnwn22xjqhcu3vd6y/app.bsky.actor.profile/self">
|
||||||
|
at://did:plc:vzsvtbtbnwn22xjqhcu3vd6y/app.bsky.actor.profile/self
|
||||||
|
</div>
|
||||||
|
<div className="at-uri" data-at-uri="at://syui.ai/app.bsky.actor.profile/self">
|
||||||
|
at://syui.ai/app.bsky.actor.profile/self
|
||||||
|
</div>
|
||||||
|
<div className="at-uri" data-at-uri="at://bsky.app/app.bsky.actor.profile/self">
|
||||||
|
at://bsky.app/app.bsky.actor.profile/self
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="instructions">
|
||||||
|
<h3>使用方法:</h3>
|
||||||
|
<ol>
|
||||||
|
<li>上記のAT URIをクリックしてください</li>
|
||||||
|
<li>モーダルがポップアップし、AT Protocolレコードの内容が表示されます</li>
|
||||||
|
<li>モーダルは×ボタンまたはEscキーで閉じることができます</li>
|
||||||
|
<li>モーダルはレスポンシブ対応で、異なる画面サイズに対応します</li>
|
||||||
|
</ol>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="test-section">
|
||||||
|
<h2>AT URI について</h2>
|
||||||
|
<p>AT URIは、AT Protocolで使用される統一リソース識別子です。この形式により、分散ソーシャルネットワーク上のコンテンツを一意に識別できます。</p>
|
||||||
|
<p>このブラウザを使用することで、ブログ投稿やその他のコンテンツに埋め込まれたAT URIを直接探索することが可能です。</p>
|
||||||
|
|
||||||
|
<h3>対応PDS環境</h3>
|
||||||
|
<ul>
|
||||||
|
<li><strong>bsky.social</strong> - メインのBlueskyネットワーク</li>
|
||||||
|
<li><strong>syu.is</strong> - 独立したPDS環境</li>
|
||||||
|
<li><strong>plc.directory</strong> + <strong>plc.syu.is</strong> - DID解決</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p><small>注意: 独立したPDS環境では、レコードの同期状況により、一部のコンテンツが利用できない場合があります。</small></p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<a href="/" className="back-link">← ブログに戻る</a>
|
||||||
|
</div>
|
||||||
|
</AtUriBrowser>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default App
|
75
pds/src/components/AtUriBrowser.jsx
Normal file
75
pds/src/components/AtUriBrowser.jsx
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
/*
|
||||||
|
* AT URI Browser Component
|
||||||
|
* Copyright (c) 2025 ai.log
|
||||||
|
* MIT License
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React, { useState, useEffect } from 'react'
|
||||||
|
import { AtUriModal } from './AtUriModal.jsx'
|
||||||
|
import { isAtUri } from '../lib/atproto.js'
|
||||||
|
|
||||||
|
export function AtUriBrowser({ children }) {
|
||||||
|
const [modalUri, setModalUri] = useState(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleAtUriClick = (e) => {
|
||||||
|
const target = e.target
|
||||||
|
|
||||||
|
// Check if clicked element has at-uri data attribute
|
||||||
|
if (target.dataset.atUri) {
|
||||||
|
e.preventDefault()
|
||||||
|
setModalUri(target.dataset.atUri)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if clicked element contains at-uri text
|
||||||
|
const text = target.textContent
|
||||||
|
if (text && isAtUri(text)) {
|
||||||
|
e.preventDefault()
|
||||||
|
setModalUri(text)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if parent element has at-uri
|
||||||
|
const parent = target.parentElement
|
||||||
|
if (parent && parent.dataset.atUri) {
|
||||||
|
e.preventDefault()
|
||||||
|
setModalUri(parent.dataset.atUri)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('click', handleAtUriClick)
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener('click', handleAtUriClick)
|
||||||
|
}
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const handleAtUriClick = (uri) => {
|
||||||
|
setModalUri(uri)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleCloseModal = () => {
|
||||||
|
setModalUri(null)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{children}
|
||||||
|
<AtUriModal
|
||||||
|
uri={modalUri}
|
||||||
|
onClose={handleCloseModal}
|
||||||
|
onAtUriClick={handleAtUriClick}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Utility function to wrap at-uri text with clickable spans
|
||||||
|
export const wrapAtUris = (text) => {
|
||||||
|
const atUriRegex = /at:\/\/[^\s]+/g
|
||||||
|
return text.replace(atUriRegex, (match) => {
|
||||||
|
return `<span data-at-uri="${match}" style="color: blue; cursor: pointer; text-decoration: underline;">${match}</span>`
|
||||||
|
})
|
||||||
|
}
|
130
pds/src/components/AtUriJson.jsx
Normal file
130
pds/src/components/AtUriJson.jsx
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
/*
|
||||||
|
* Based on frontpage/atproto-browser
|
||||||
|
* Copyright (c) 2025 The Frontpage Authors
|
||||||
|
* MIT License
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React from 'react'
|
||||||
|
import { isDid } from '@atproto/did'
|
||||||
|
import { parseAtUri, isAtUri } from '../lib/atproto.js'
|
||||||
|
|
||||||
|
const JSONString = ({ data, onAtUriClick }) => {
|
||||||
|
const handleClick = (uri) => {
|
||||||
|
if (onAtUriClick) {
|
||||||
|
onAtUriClick(uri)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<pre style={{ color: 'darkgreen', margin: 0, display: 'inline' }}>
|
||||||
|
{isAtUri(data) ? (
|
||||||
|
<>
|
||||||
|
"
|
||||||
|
<span
|
||||||
|
onClick={() => handleClick(data)}
|
||||||
|
style={{
|
||||||
|
color: 'blue',
|
||||||
|
cursor: 'pointer',
|
||||||
|
textDecoration: 'underline'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{data}
|
||||||
|
</span>
|
||||||
|
"
|
||||||
|
</>
|
||||||
|
) : isDid(data) ? (
|
||||||
|
<>
|
||||||
|
"
|
||||||
|
<span
|
||||||
|
onClick={() => handleClick(`at://${data}`)}
|
||||||
|
style={{
|
||||||
|
color: 'blue',
|
||||||
|
cursor: 'pointer',
|
||||||
|
textDecoration: 'underline'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{data}
|
||||||
|
</span>
|
||||||
|
"
|
||||||
|
</>
|
||||||
|
) : URL.canParse(data) ? (
|
||||||
|
<>
|
||||||
|
"
|
||||||
|
<a href={data} rel="noopener noreferrer ugc" target="_blank">
|
||||||
|
{data}
|
||||||
|
</a>
|
||||||
|
"
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
`"${data}"`
|
||||||
|
)}
|
||||||
|
</pre>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const JSONValue = ({ data, onAtUriClick }) => {
|
||||||
|
if (data === null) {
|
||||||
|
return <pre style={{ color: 'gray', margin: 0, display: 'inline' }}>null</pre>
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof data === 'string') {
|
||||||
|
return <JSONString data={data} onAtUriClick={onAtUriClick} />
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof data === 'number') {
|
||||||
|
return <pre style={{ color: 'darkorange', margin: 0, display: 'inline' }}>{data}</pre>
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof data === 'boolean') {
|
||||||
|
return <pre style={{ color: 'darkred', margin: 0, display: 'inline' }}>{data.toString()}</pre>
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(data)) {
|
||||||
|
return (
|
||||||
|
<div style={{ paddingLeft: '20px' }}>
|
||||||
|
[
|
||||||
|
{data.map((item, index) => (
|
||||||
|
<div key={index} style={{ paddingLeft: '20px' }}>
|
||||||
|
<JSONValue data={item} onAtUriClick={onAtUriClick} />
|
||||||
|
{index < data.length - 1 && ','}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
]
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof data === 'object') {
|
||||||
|
return (
|
||||||
|
<div style={{ paddingLeft: '20px' }}>
|
||||||
|
{'{'}
|
||||||
|
{Object.entries(data).map(([key, value], index, entries) => (
|
||||||
|
<div key={key} style={{ paddingLeft: '20px' }}>
|
||||||
|
<span style={{ color: 'darkblue' }}>"{key}"</span>: <JSONValue data={value} onAtUriClick={onAtUriClick} />
|
||||||
|
{index < entries.length - 1 && ','}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{'}'}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return <pre style={{ margin: 0, display: 'inline' }}>{String(data)}</pre>
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function AtUriJson({ data, onAtUriClick }) {
|
||||||
|
return (
|
||||||
|
<div style={{
|
||||||
|
fontFamily: 'monospace',
|
||||||
|
fontSize: '14px',
|
||||||
|
padding: '10px',
|
||||||
|
backgroundColor: '#f5f5f5',
|
||||||
|
border: '1px solid #ddd',
|
||||||
|
borderRadius: '4px',
|
||||||
|
overflow: 'auto',
|
||||||
|
maxHeight: '400px'
|
||||||
|
}}>
|
||||||
|
<JSONValue data={data} onAtUriClick={onAtUriClick} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
80
pds/src/components/AtUriModal.jsx
Normal file
80
pds/src/components/AtUriModal.jsx
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
/*
|
||||||
|
* AT URI Modal Component
|
||||||
|
* Copyright (c) 2025 ai.log
|
||||||
|
* MIT License
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React, { useEffect } from 'react'
|
||||||
|
import AtUriViewer from './AtUriViewer.jsx'
|
||||||
|
|
||||||
|
export function AtUriModal({ uri, onClose, onAtUriClick }) {
|
||||||
|
useEffect(() => {
|
||||||
|
const handleEscape = (e) => {
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
onClose()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleClickOutside = (e) => {
|
||||||
|
if (e.target.classList.contains('at-uri-modal-overlay')) {
|
||||||
|
onClose()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('keydown', handleEscape)
|
||||||
|
document.addEventListener('click', handleClickOutside)
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener('keydown', handleEscape)
|
||||||
|
document.removeEventListener('click', handleClickOutside)
|
||||||
|
}
|
||||||
|
}, [onClose])
|
||||||
|
|
||||||
|
if (!uri) return null
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="at-uri-modal-overlay" style={{
|
||||||
|
position: 'fixed',
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
bottom: 0,
|
||||||
|
backgroundColor: 'rgba(0, 0, 0, 0.5)',
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
zIndex: 1000
|
||||||
|
}}>
|
||||||
|
<div style={{
|
||||||
|
backgroundColor: 'white',
|
||||||
|
borderRadius: '8px',
|
||||||
|
maxWidth: '800px',
|
||||||
|
maxHeight: '600px',
|
||||||
|
width: '90%',
|
||||||
|
height: '80%',
|
||||||
|
overflow: 'auto',
|
||||||
|
position: 'relative',
|
||||||
|
boxShadow: '0 4px 6px rgba(0, 0, 0, 0.1)'
|
||||||
|
}}>
|
||||||
|
<button
|
||||||
|
onClick={onClose}
|
||||||
|
style={{
|
||||||
|
position: 'absolute',
|
||||||
|
top: '10px',
|
||||||
|
right: '10px',
|
||||||
|
background: 'none',
|
||||||
|
border: 'none',
|
||||||
|
fontSize: '20px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
zIndex: 1001,
|
||||||
|
padding: '5px 10px'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
×
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<AtUriViewer uri={uri} onAtUriClick={onAtUriClick} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
103
pds/src/components/AtUriViewer.jsx
Normal file
103
pds/src/components/AtUriViewer.jsx
Normal file
@@ -0,0 +1,103 @@
|
|||||||
|
/*
|
||||||
|
* Based on frontpage/atproto-browser
|
||||||
|
* Copyright (c) 2025 The Frontpage Authors
|
||||||
|
* MIT License
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React, { useState, useEffect } from 'react'
|
||||||
|
import { parseAtUri, getRecord } from '../lib/atproto.js'
|
||||||
|
import AtUriJson from './AtUriJson.jsx'
|
||||||
|
|
||||||
|
export default function AtUriViewer({ uri, onAtUriClick }) {
|
||||||
|
const [record, setRecord] = useState(null)
|
||||||
|
const [loading, setLoading] = useState(true)
|
||||||
|
const [error, setError] = useState(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const loadRecord = async () => {
|
||||||
|
if (!uri) return
|
||||||
|
|
||||||
|
setLoading(true)
|
||||||
|
setError(null)
|
||||||
|
|
||||||
|
try {
|
||||||
|
const atUri = parseAtUri(uri)
|
||||||
|
if (!atUri) {
|
||||||
|
throw new Error('Invalid AT URI')
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const result = await getRecord(atUri.hostname, atUri.collection, atUri.rkey)
|
||||||
|
|
||||||
|
|
||||||
|
if (!result.success) {
|
||||||
|
throw new Error(result.error)
|
||||||
|
}
|
||||||
|
|
||||||
|
setRecord(result.data)
|
||||||
|
} catch (err) {
|
||||||
|
setError(err.message)
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
loadRecord()
|
||||||
|
}, [uri])
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return (
|
||||||
|
<div style={{ padding: '20px', textAlign: 'center' }}>
|
||||||
|
<div>Loading...</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
return (
|
||||||
|
<div style={{ padding: '20px', color: 'red' }}>
|
||||||
|
<div><strong>Error:</strong> {error}</div>
|
||||||
|
<div style={{ marginTop: '10px', fontSize: '12px' }}>
|
||||||
|
<strong>URI:</strong> {uri}
|
||||||
|
</div>
|
||||||
|
<div style={{ marginTop: '10px', fontSize: '12px', color: '#666' }}>
|
||||||
|
デバッグ情報: このAT URIは有効ではないか、レコードが存在しません。
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!record) {
|
||||||
|
return (
|
||||||
|
<div style={{ padding: '20px' }}>
|
||||||
|
<div>No record found</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const atUri = parseAtUri(uri)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={{ padding: '20px' }}>
|
||||||
|
<div style={{ marginBottom: '20px' }}>
|
||||||
|
<h3 style={{ margin: '0 0 10px 0', fontSize: '18px' }}>AT URI Record</h3>
|
||||||
|
<div style={{
|
||||||
|
fontSize: '14px',
|
||||||
|
color: '#666',
|
||||||
|
fontFamily: 'monospace',
|
||||||
|
wordBreak: 'break-all'
|
||||||
|
}}>
|
||||||
|
{uri}
|
||||||
|
</div>
|
||||||
|
<div style={{ fontSize: '12px', color: '#999', marginTop: '5px' }}>
|
||||||
|
DID: {atUri.hostname} | Collection: {atUri.collection} | RKey: {atUri.rkey}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<h4 style={{ margin: '0 0 10px 0', fontSize: '16px' }}>Record Data</h4>
|
||||||
|
<AtUriJson data={record} onAtUriClick={onAtUriClick} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
33
pds/src/config.js
Normal file
33
pds/src/config.js
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
* AT Protocol Configuration for syu.is environment
|
||||||
|
*/
|
||||||
|
|
||||||
|
export const AT_PROTOCOL_CONFIG = {
|
||||||
|
// Primary PDS environment (syu.is)
|
||||||
|
primary: {
|
||||||
|
pds: 'https://syu.is',
|
||||||
|
plc: 'https://plc.syu.is',
|
||||||
|
bsky: 'https://bsky.syu.is',
|
||||||
|
web: 'https://web.syu.is'
|
||||||
|
},
|
||||||
|
|
||||||
|
// Fallback PDS environment (bsky.social)
|
||||||
|
fallback: {
|
||||||
|
pds: 'https://bsky.social',
|
||||||
|
plc: 'https://plc.directory',
|
||||||
|
bsky: 'https://public.api.bsky.app',
|
||||||
|
web: 'https://bsky.app'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getPDSConfig = (pds) => {
|
||||||
|
// Map PDS URL to appropriate config
|
||||||
|
if (pds.includes('syu.is')) {
|
||||||
|
return AT_PROTOCOL_CONFIG.primary
|
||||||
|
} else if (pds.includes('bsky.social')) {
|
||||||
|
return AT_PROTOCOL_CONFIG.fallback
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default to primary for unknown PDS
|
||||||
|
return AT_PROTOCOL_CONFIG.primary
|
||||||
|
}
|
9
pds/src/index.js
Normal file
9
pds/src/index.js
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
/*
|
||||||
|
* Based on frontpage/atproto-browser
|
||||||
|
* Copyright (c) 2025 The Frontpage Authors
|
||||||
|
* MIT License
|
||||||
|
*/
|
||||||
|
|
||||||
|
export { AtUriBrowser } from './components/AtUriBrowser.jsx'
|
||||||
|
export { AtUriModal } from './components/AtUriModal.jsx'
|
||||||
|
export { default as AtUriViewer } from './components/AtUriViewer.jsx'
|
155
pds/src/lib/atproto.js
Normal file
155
pds/src/lib/atproto.js
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
/*
|
||||||
|
* Based on frontpage/atproto-browser
|
||||||
|
* Copyright (c) 2025 The Frontpage Authors
|
||||||
|
* MIT License
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { AtpBaseClient } from '@atproto/api'
|
||||||
|
import { AtUri } from '@atproto/syntax'
|
||||||
|
import { isDid } from '@atproto/did'
|
||||||
|
import { AT_PROTOCOL_CONFIG } from '../config.js'
|
||||||
|
|
||||||
|
// Identity resolution cache
|
||||||
|
const identityCache = new Map()
|
||||||
|
|
||||||
|
// Create AT Protocol client
|
||||||
|
export const createAtpClient = (pds) => {
|
||||||
|
return new AtpBaseClient({
|
||||||
|
service: pds.startsWith('http') ? pds : `https://${pds}`
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resolve identity (DID/Handle)
|
||||||
|
export const resolveIdentity = async (identifier) => {
|
||||||
|
if (identityCache.has(identifier)) {
|
||||||
|
return identityCache.get(identifier)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
let did = identifier
|
||||||
|
|
||||||
|
// If it's a handle, resolve to DID
|
||||||
|
if (!isDid(identifier)) {
|
||||||
|
// Try syu.is first, then fallback to bsky.social
|
||||||
|
let resolved = false
|
||||||
|
|
||||||
|
try {
|
||||||
|
const client = createAtpClient(AT_PROTOCOL_CONFIG.primary.pds)
|
||||||
|
const response = await client.com.atproto.repo.describeRepo({ repo: identifier })
|
||||||
|
did = response.data.did
|
||||||
|
resolved = true
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!resolved) {
|
||||||
|
try {
|
||||||
|
const client = createAtpClient(AT_PROTOCOL_CONFIG.fallback.pds)
|
||||||
|
const response = await client.com.atproto.repo.describeRepo({ repo: identifier })
|
||||||
|
did = response.data.did
|
||||||
|
} catch (error) {
|
||||||
|
throw new Error(`Failed to resolve handle: ${identifier}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get DID document to find PDS
|
||||||
|
// Try plc.syu.is first, then fallback to plc.directory
|
||||||
|
let didDoc = null
|
||||||
|
let plcResponse = null
|
||||||
|
|
||||||
|
try {
|
||||||
|
plcResponse = await fetch(`${AT_PROTOCOL_CONFIG.primary.plc}/${did}`)
|
||||||
|
if (plcResponse.ok) {
|
||||||
|
didDoc = await plcResponse.json()
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
|
||||||
|
// If plc.syu.is fails, try plc.directory
|
||||||
|
if (!didDoc) {
|
||||||
|
try {
|
||||||
|
plcResponse = await fetch(`${AT_PROTOCOL_CONFIG.fallback.plc}/${did}`)
|
||||||
|
if (plcResponse.ok) {
|
||||||
|
didDoc = await plcResponse.json()
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!didDoc) {
|
||||||
|
throw new Error(`Failed to resolve DID document from any PLC server`)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find PDS service endpoint
|
||||||
|
const pdsService = didDoc.service?.find(service =>
|
||||||
|
service.type === 'AtprotoPersonalDataServer' ||
|
||||||
|
service.id === '#atproto_pds'
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!pdsService) {
|
||||||
|
throw new Error('No PDS service found in DID document')
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = {
|
||||||
|
success: true,
|
||||||
|
didDocument: didDoc,
|
||||||
|
pdsUrl: pdsService.serviceEndpoint
|
||||||
|
}
|
||||||
|
|
||||||
|
identityCache.set(identifier, result)
|
||||||
|
return result
|
||||||
|
} catch (error) {
|
||||||
|
const result = {
|
||||||
|
success: false,
|
||||||
|
error: error.message
|
||||||
|
}
|
||||||
|
identityCache.set(identifier, result)
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get record from AT Protocol
|
||||||
|
export const getRecord = async (did, collection, rkey) => {
|
||||||
|
try {
|
||||||
|
const identityResult = await resolveIdentity(did)
|
||||||
|
|
||||||
|
if (!identityResult.success) {
|
||||||
|
return { success: false, error: identityResult.error }
|
||||||
|
}
|
||||||
|
|
||||||
|
const pdsUrl = identityResult.pdsUrl
|
||||||
|
|
||||||
|
const client = createAtpClient(pdsUrl)
|
||||||
|
|
||||||
|
const response = await client.com.atproto.repo.getRecord({
|
||||||
|
repo: did,
|
||||||
|
collection,
|
||||||
|
rkey
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
data: response.data,
|
||||||
|
pdsUrl
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
error: error.message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse AT URI
|
||||||
|
export const parseAtUri = (uri) => {
|
||||||
|
try {
|
||||||
|
return new AtUri(uri)
|
||||||
|
} catch (error) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if string is AT URI
|
||||||
|
export const isAtUri = (str) => {
|
||||||
|
return str.startsWith('at://') && str.split(' ').length === 1
|
||||||
|
}
|
9
pds/src/main.jsx
Normal file
9
pds/src/main.jsx
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import React from 'react'
|
||||||
|
import ReactDOM from 'react-dom/client'
|
||||||
|
import App from './App.jsx'
|
||||||
|
|
||||||
|
ReactDOM.createRoot(document.getElementById('root')).render(
|
||||||
|
<React.StrictMode>
|
||||||
|
<App />
|
||||||
|
</React.StrictMode>,
|
||||||
|
)
|
10
pds/vite.config.js
Normal file
10
pds/vite.config.js
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import { defineConfig } from 'vite'
|
||||||
|
import react from '@vitejs/plugin-react'
|
||||||
|
|
||||||
|
export default defineConfig({
|
||||||
|
plugins: [react()],
|
||||||
|
base: '/pds/',
|
||||||
|
define: {
|
||||||
|
'process.env.NODE_ENV': JSON.stringify('production')
|
||||||
|
}
|
||||||
|
})
|
22
scpt/run.zsh
22
scpt/run.zsh
@@ -5,6 +5,7 @@ function _env() {
|
|||||||
ailog=$d/target/release/ailog
|
ailog=$d/target/release/ailog
|
||||||
oauth=$d/oauth
|
oauth=$d/oauth
|
||||||
myblog=$d/my-blog
|
myblog=$d/my-blog
|
||||||
|
pds=$d/pds
|
||||||
port=4173
|
port=4173
|
||||||
#source $oauth/.env.production
|
#source $oauth/.env.production
|
||||||
case $OSTYPE in
|
case $OSTYPE in
|
||||||
@@ -43,6 +44,21 @@ function _oauth_build() {
|
|||||||
#npm run preview
|
#npm run preview
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function _pds_build() {
|
||||||
|
cd $pds
|
||||||
|
nvm use 21
|
||||||
|
npm i
|
||||||
|
npm run build
|
||||||
|
rm -rf $myblog/static/pds
|
||||||
|
cp -rf dist $myblog/static/pds
|
||||||
|
}
|
||||||
|
|
||||||
|
function _pds_server() {
|
||||||
|
cd $pds
|
||||||
|
nvm use 21
|
||||||
|
npm run preview
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function _server_comment() {
|
function _server_comment() {
|
||||||
cargo build --release
|
cargo build --release
|
||||||
@@ -65,6 +81,12 @@ case "${1:-serve}" in
|
|||||||
oauth|o)
|
oauth|o)
|
||||||
_oauth_build
|
_oauth_build
|
||||||
;;
|
;;
|
||||||
|
pds|p)
|
||||||
|
_pds_build
|
||||||
|
;;
|
||||||
|
pds-server|ps)
|
||||||
|
_pds_server
|
||||||
|
;;
|
||||||
n)
|
n)
|
||||||
oauth=$d/oauth_old
|
oauth=$d/oauth_old
|
||||||
_oauth_build
|
_oauth_build
|
||||||
|
@@ -328,7 +328,7 @@ async fn serve_file(path: &str) -> Result<(&'static str, Vec<u8>, &'static str)>
|
|||||||
// Remove query parameters from path
|
// Remove query parameters from path
|
||||||
let clean_path = path.split('?').next().unwrap_or(path);
|
let clean_path = path.split('?').next().unwrap_or(path);
|
||||||
|
|
||||||
let file_path = if clean_path == "/" {
|
let mut file_path = if clean_path == "/" {
|
||||||
PathBuf::from("public/index.html")
|
PathBuf::from("public/index.html")
|
||||||
} else {
|
} else {
|
||||||
PathBuf::from("public").join(clean_path.trim_start_matches('/'))
|
PathBuf::from("public").join(clean_path.trim_start_matches('/'))
|
||||||
@@ -337,9 +337,42 @@ async fn serve_file(path: &str) -> Result<(&'static str, Vec<u8>, &'static str)>
|
|||||||
println!("Serving file: {}", file_path.display());
|
println!("Serving file: {}", file_path.display());
|
||||||
|
|
||||||
// Check if file exists and get metadata
|
// Check if file exists and get metadata
|
||||||
let metadata = tokio::fs::metadata(&file_path).await?;
|
let metadata = tokio::fs::metadata(&file_path).await;
|
||||||
if !metadata.is_file() {
|
|
||||||
return Err(anyhow::anyhow!("Not a file: {}", file_path.display()));
|
match metadata {
|
||||||
|
Ok(meta) if meta.is_file() => {
|
||||||
|
// File exists, proceed normally
|
||||||
|
}
|
||||||
|
Ok(meta) if meta.is_dir() => {
|
||||||
|
// Directory exists, try to serve index.html
|
||||||
|
file_path = file_path.join("index.html");
|
||||||
|
println!("Directory found, trying index.html: {}", file_path.display());
|
||||||
|
let index_metadata = tokio::fs::metadata(&file_path).await?;
|
||||||
|
if !index_metadata.is_file() {
|
||||||
|
return Err(anyhow::anyhow!("No index.html in directory: {}", file_path.display()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(_) => {
|
||||||
|
return Err(anyhow::anyhow!("Not a file: {}", file_path.display()));
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
// Try adding index.html to the original path
|
||||||
|
let index_path = PathBuf::from("public")
|
||||||
|
.join(clean_path.trim_start_matches('/'))
|
||||||
|
.join("index.html");
|
||||||
|
|
||||||
|
println!("File not found, trying index.html: {}", index_path.display());
|
||||||
|
let index_metadata = tokio::fs::metadata(&index_path).await;
|
||||||
|
if let Ok(meta) = index_metadata {
|
||||||
|
if meta.is_file() {
|
||||||
|
file_path = index_path;
|
||||||
|
} else {
|
||||||
|
return Err(anyhow::anyhow!("Original error: {}", e));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return Err(anyhow::anyhow!("File not found: {}", file_path.display()));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let (content_type, cache_control) = match file_path.extension().and_then(|ext| ext.to_str()) {
|
let (content_type, cache_control) = match file_path.extension().and_then(|ext| ext.to_str()) {
|
||||||
|
@@ -86,6 +86,12 @@ impl Generator {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Generate PDS page
|
||||||
|
self.generate_pds_page().await?;
|
||||||
|
|
||||||
|
// Generate Game page
|
||||||
|
self.generate_game_page().await?;
|
||||||
|
|
||||||
println!("{} {} posts", "Generated".cyan(), posts.len());
|
println!("{} {} posts", "Generated".cyan(), posts.len());
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -491,6 +497,54 @@ impl Generator {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn generate_pds_page(&self) -> Result<()> {
|
||||||
|
let public_dir = self.base_path.join("public");
|
||||||
|
let pds_dir = public_dir.join("pds");
|
||||||
|
fs::create_dir_all(&pds_dir)?;
|
||||||
|
|
||||||
|
// Generate PDS page using the pds.html template
|
||||||
|
let config_with_timestamp = self.create_config_with_timestamp()?;
|
||||||
|
let mut context = tera::Context::new();
|
||||||
|
context.insert("config", &config_with_timestamp);
|
||||||
|
context.insert("site", &self.config.site);
|
||||||
|
context.insert("page", &serde_json::json!({
|
||||||
|
"title": "AT URI Browser",
|
||||||
|
"description": "AT Protocol レコードをブラウズし、分散SNSのコンテンツを探索できます"
|
||||||
|
}));
|
||||||
|
|
||||||
|
let rendered_content = self.template_engine.render("pds.html", &context)?;
|
||||||
|
let output_path = pds_dir.join("index.html");
|
||||||
|
fs::write(output_path, rendered_content)?;
|
||||||
|
|
||||||
|
println!("{} PDS page", "Generated".cyan());
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn generate_game_page(&self) -> Result<()> {
|
||||||
|
let public_dir = self.base_path.join("public");
|
||||||
|
let game_dir = public_dir.join("game");
|
||||||
|
fs::create_dir_all(&game_dir)?;
|
||||||
|
|
||||||
|
// Generate Game page using the game.html template
|
||||||
|
let config_with_timestamp = self.create_config_with_timestamp()?;
|
||||||
|
let mut context = tera::Context::new();
|
||||||
|
context.insert("config", &config_with_timestamp);
|
||||||
|
context.insert("site", &self.config.site);
|
||||||
|
context.insert("page", &serde_json::json!({
|
||||||
|
"title": "Game",
|
||||||
|
"description": "Play the game with AT Protocol authentication"
|
||||||
|
}));
|
||||||
|
|
||||||
|
let rendered_content = self.template_engine.render("game.html", &context)?;
|
||||||
|
let output_path = game_dir.join("index.html");
|
||||||
|
fs::write(output_path, rendered_content)?;
|
||||||
|
|
||||||
|
println!("{} Game page", "Generated".cyan());
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
fn extract_plain_text(&self, html_content: &str) -> String {
|
fn extract_plain_text(&self, html_content: &str) -> String {
|
||||||
// Remove HTML tags and extract plain text
|
// Remove HTML tags and extract plain text
|
||||||
let mut text = String::new();
|
let mut text = String::new();
|
||||||
@@ -536,6 +590,7 @@ pub struct Post {
|
|||||||
pub extra: Option<serde_json::Value>,
|
pub extra: Option<serde_json::Value>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, serde::Serialize)]
|
#[derive(Debug, Clone, serde::Serialize)]
|
||||||
pub struct Translation {
|
pub struct Translation {
|
||||||
pub lang: String,
|
pub lang: String,
|
||||||
|
Reference in New Issue
Block a user