- 必要環境
- はじめに
- ディレクトリ構造
- 複数のウェブサイトを管理する方法
- ページ
- フロントマター
- 配色
- スタイルシート
- JavaScriptファイル
- ファビコン
- 画像
- アニメーション
- 音声
- フォント
- シンボル
- レイアウト
- ラッパー
- セグメント
- 部品
- アーティクル
- 何が何に埋め込めるのか?
- リンク
- リンクリスト
- 動的要素
- ミニアプリの埋め込み
- プラグイン
- サイトプロパティとプロパティの継承
- head要素の内容の管理
- Cloudflare Pagesへのデプロイメント
- 特殊なタグと属性のリスト
- TODO List
- License
- Node.js: 20 以上
- npm: 8.0 以上
まず最初に、tgweb 用の「作業ディレクトリ」を作成します。ディレクトリの場所はどこでもかまいません。
例として、ホームディレクトリの下層に my_site
というサブディレクトリを作成し、作業ディレクトリとして利用します。
mkdir -p ~/my_site
そして作業ディレクトリに移動し、npm
コマンドで tgweb をインストールします。
cd my_site
npm install tgweb@latest
インストールが無事に終わると、作業ディレクトリの中に node_modules
というディレクトリが作成され、このディレクトリの中に tgweb とその依存関係にあるライブラリがインストールされます。
tgweb を使用してウェブサイト作成を始めるにあたり、以下のコマンドを作業ディレクトリで実行します。
npx tgweb-init
このコマンドにより、src
と dist
という 2 つのサブディレクトリといくつかのファイルが作成されます。
my_site/src/pages
ディレクトリにある index.html
の初期状態の内容は以下になります。
<body>
<p class="p-2 text-red-500">Hello, world</p>
</body>
body
要素が <html>
</html>
タグで囲われていないことにご注意ください。
tgweb サーバを起動するには、以下のコマンドを実行します。
npx tgweb-server
このコマンドは、指定がなければ tgweb サーバをポート 3000 で公開するように起動します。 すでにポート 3000 が使用中であれば、以下のエラーメッセージが表示されます。
ERROR: Could not start a web server. Port 3000 is in use.
他のポートを指定したい場合は、環境変数 PORT
で以下のように指定します。
PORT=4000 npx tgweb-server
tgweb サーバーが正常に起動すると、以下のメッセージが表示されます。
tailwindcss began to monitor the HTML files for changes.
Web server is listening on port 3000.
Rebuilding tailwind.css. Done in 761ms.
http://loclahost:3000
をブラウザで開いてください。
"Hello, world!" というテキストが赤い文字色で表示されることを確認してください。
<p>
タグの class
属性に指定されている text-red-500
と p-2
クラストークンは Tailwind CSS を利用するための記述です。
text-red-500
トークンは <p>
タグで囲んだ要素の文字色に coral red (#ef4444
) を指定します。
p-2
トークンは、<p>
タグのパディングに 2 (8px/0.5rem) を指定します。
dist
ディレクトリ内に index.html
が生成されていることを確認してください。
このファイルは、<html>
や <head>
要素を含む完全な HTML 文書になっています。
tgweb は、さらに Tailwind CSS の補助として my_site/dist/css
ディレクトリの中に tailwind.css
というファイルを生成します。
初期状態では、my_site/dist/index.html
内の <title>
要素が "No Title" と設定されています。この箇所については後ほど取り上げます。
my_site/src/pages/index.html
の内容を以下のように変更します。
<body>
<p class="p-2 text-green-500">Hello, world</p>
</body>
http://localhost:3000
を表示しているブラウザが自動的に再読込され、"Hello, world!" の文字色が赤から緑に変更されます。
ターミナルで Ctrl + C
キーを押すと tgweb サーバが停止します。
配布用ウェブサイトデータを生成するには、以下のコマンドを実行します。
npx tgweb-dist
npx tgweb-init
コマンドの実行によって以下のディレクトリ構造が生成されます。
.
├── dist
├── node_modules
├── src
│ ├── animations
│ ├── articles
│ ├── audios
│ ├── css
│ ├── components
│ ├── images
│ ├── js
│ ├── layouts
│ ├── pages
│ │ └── index.html
│ ├── segments
│ ├── shared_components
│ ├── shared_css
│ ├── shared_js
│ ├── shared_wrappers
│ ├── tags
│ ├── color_scheme.toml
│ └── site.toml
├── package-lock.json
└── package.json
以下は特記事項です。
- tgweb は
src
ディレクトリ内を走査し、HTML ファイルをdist
ディレクトリ内に生成します。 - tgweb は
src
ディレクトリ内を走査し、画像ファイルと音声ファイルをdist
ディレクトリ内に複製します。 - Tailwind CSS は、
dist
ディレクトリ内の HTML ファイルを走査し、my_site/dist/css
ディレクトリ内にtailwind.css
という名前の CSS ファイルを生成します。 - 上記の通りファイルが自動生成されるため、
my_site/dist
ディレクトリ内のファイルを手動で変更しても上書きにより失われます。 - ユーザが独自の CSS ファイルを追加することはできません。
- ユーザが独自の JavaScript ファイルを追加することはできません。
はじめに では、tgweb を作業ディレクトリにインストールし、ひとつのウェブサイト構造を扱いました。 しかし、同一の作業ディレクトリ以下で複数のウェブサイト構造を構築することも可能です。
ここでは例として、ユーザのホームディレクトリの下層に web
という作業ディレクトリがあると想定します。
tgweb を npm
経由でインストールする時点までは、ひとつのウェブサイト構造を構築する場合と手順は同じです。
mkdir ~/web
cd web
npm install tgweb@latest
作業ディレクトリ内に最初に構築するウェブサイトディレクトリの名前を決めます。
例として、site_0
と名付けます。次に以下のコマンドを実行します。
npx tgweb-init site_0
このコマンドを実行することにより sites
サブディレクトリが生成され、更にその中に site_0
ディレクトリが生成されます。
site_0
ディレクトリの中には、src
、dist
ディレクトリといくつかのファイルが生成されます。
作業ディレクトリ内に sites
ディレクトリがある場合を、「複数サイト構造」とよび、
sites
ディレクトリがない場合を「単一サイト構造」とよびます。
上記のコマンドにより、tgweb は web/sites/site_0/src/pages
ディレクトリ内に index.html
を以下の内容で生成します。
<body>
<p class="p-2 text-red-500">Hello, world</p>
</body>
tgweb サーバの起動は以下のコマンドを実行します。
npx tgweb-server site_0
対象のウェブサイトのディレクトリ名を npx tgweb-init
や npx tgweb-server
の引数に渡すことで
ひとつの作業ディレクトリ内でも複数のウェブサイトを管理できます。
複数サイト構造を採用した場合、作業ディレクトリの構造は例えば以下のようになります。
.
├── node_modules
├── package-lock.json
├── package.json
└── sites
├── site_0
│ ├── dist
│ └── src
│ ├── animations
│ ├── articles
│ ├── audios
│ ├── components
│ ├── css
│ ├── images
│ ├── js
│ ├── layouts
│ ├── pages
│ │ └── index.html
│ ├── segments
│ ├── shared_components
│ ├── shared_css
│ ├── shared_js
│ ├── shared_wrappers
│ ├── tags
│ ├── color_scheme.toml
│ └── site.toml
└── site_1
├── dist
└── src
├── animations
├── articles
├── audios
├── components
├── css
├── images
├── js
├── layouts
├── pages
│ └── index.html
├── segments
├── shared_components
├── shared_css
├── shared_js
├── shared_wrappers
├── tags
├── color_scheme.toml
└── site.toml
sites/
を含んだ表記でディレクトリ名を tgweb サーバのコマンド引数に渡すこともできます。
npx tgweb-server sites/site_0
コマンドを打ち込む文字数が増えますが、シェル補完によって末尾の番号を変更することが容易になります。
サブディレクトリ sites/_site_0
にあるウェブサイトの配布用データを生刺青するには、以下のコマンドを実行します。
npx tgweb-dist site_0
単一サイト構造から複数サイト構造に作業ディレクトリの構造を変える場合は、以下のコマンドを実行します。
site_0
のディレクトリ名は、適切なものに変更してください。
mkdir -p sites/site_0
mv dist src tailwind.* sites/site_0
反対に複数サイト構造から単一サイト構造に作業ディレクトリの構造を変える場合は、以下のコマンドを実行してください。
mv sites/site_0/* .
rm -rf sites
注意点として、上記のコマンドは、sites
下層のディレクトリに複数のディレクトリがあった場合 site_0
以外のディレクトリを削除します。
npx tgweb-archive
コマンドを使用すると src
ディレクトリをZIP形式で圧縮したファイルを作ることができます。
単一サイト構造の場合、作業ディレクトリ上でこのコマンドを実行すると作業ディレクトリの名前に拡張子 .zip
を加えた名前のアーカイブファイルがカレントディレクトリに作られます。
複数サイト構造の場合、作業ディレクトリ上で npx tgweb-archive site_0
のように対象ウェブサイトのディレクトリ名を指定してください。その結果、そのディレクトリ名に拡張子 .zip
を加えた名前のアーカイブファイルがカレントディレクトリ作られます。
tgweb では、ウェブサイトを構成するHTML文書は、テンプレートファイルの組み合わせから生成されます。「ページ」はそのようなテンプレート・ファイルの一種です。
ページ以外のテンプレートには、レイアウト、セグメント、ラッパー、アーティクル、部品があり、以降で順番に説明します。
ページは作業ディレクトリ下のsrc/pages
サブディレクトリに置かれます。
src/pages
ディレクトリの下にサブディレクトリを作り、その下にページを置くことも可能です。
ただし、src/pages
ディレクトリ直下に以下のような名前のサブディレクトリを作ることはできません:
animations
articles
audios
images
tags
videos
すべてのウェブサイトは、src/pages
ディレクトリにindex.html
という名前のファイルを持たなければなりません。
このページから、ウェブサイトの「ホームページ」が作成されます。
レイアウトが適用されていないページは「シンプルページ」と呼ばれます。
次はシンプルページの例です:
<body>
<h1 class="text-2xl font-bold">Greeting</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</body>
ページのトップレベル要素は <body>
要素でなければならないことに注意してください。
通常のHTMLファイルとは異なり、tgwebページは <html>
タグや </html>
タグで囲まれておらず、<head>
要素もありません。
ページは完全なHTMLドキュメントに変換され、dist
ディレクトリに書き込まれます。
例えば、src/pages/index.html
は dist/index.html
に、src/pages/shops/new_york.html
は dist/shops/new_york.html
に変換されます。
src/pages/index.html
を上のシンプルページの内容に変更します。
dist/index.html
には次のような <html>
や <head>
要素を含む完全なHTML文書が生成されます。
<!DOCTYPE html>
<html>
<head>
...
</head>
<body>
<body>
<h1 class="text-2xl font-bold">Greeting</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</body>
</body>
</html>
<head>
要素の内容は自動的に生成されます。
詳細は 後述 します。
ブラウザでこのページを開くには、URLとして http://localhost:3000
を指定します。
テンプレートが ---
のみで構成される行で始まり、テンプレート内にそのような行がもう1行ある場合、この2行で囲まれた領域を「フロントマターブロック」と呼びます。
この領域では、一連のプロパティに値を与えることができます。 このプロパティと値のペアのセットを「フロントマター」と呼びます。
フロントマターブロックは TOML フォーマットで記述されます。 次に示すのは、フロントマターブロックの例です:
---
[main]
title = "Our Mission"
[data]
current_year = 2023
[style]
red-box = "rounded border border-red-600 p-1 md:p-2"
# TODO: Needs to be checked by the boss.
---
上の例では、フロントマターブロックの4つの主要な構成要素が使われています。
- プロパティの定義
- セクションヘッダー
- コメント
- 空白行
#
記号で始まる行はコメントとして無視されます。空白行も無視されます。
テンプレートファイルのファイル名の本体に拡張子 .toml
を加えた名前を持つファイルが存在するとき、そのファイルの中身はテンプレートファイルのフロントマターとして解釈されます。そのファイルはフロントマターファイルと呼ばれます。
例えば、src/pages
ディレクトリに index.html
と index.toml
が存在するとき、後者は前者のフロントマターファイルとなります。
テンプレートファイルがフロントマターブロックとフロントマターファイルの両方を持つ場合は、フロントマターファイルが優先されます。
title = "Our Mission"
はプロパティ定義の例です。
等号の左側の title
がプロパティの名前で、右側の "Our Mission"
がプロパティの値です。
プロパティ名、等号、値は同じ行に記述する必要がありますが、値によっては複数行にまたがるものもあります。
プロパティ名が大文字と小文字、数字、アンダースコア、マイナス記号のみで構成されている場合は、引用符なしで書くことができます。; そうでない場合は、次の例のように引用符で囲む必要がある。
"&_p" = "mb-2"
プロパティの値は、その型によって異なる方法で記述します。
文字列は常に引用符で囲む必要があります。
100
、-16
、3.14
のように、整数と浮動小数点数は引用符で囲みません。
真偽値(true
と false
)も引用符で囲まず、小文字で記述します。
その他の書き方については、例が出てきたときに説明します。
先の例で、[data]
と [style]
と書かれた行を「セクションヘッダー」と呼びます。TOMLでは「テーブルヘッダー」と呼びますが、Tgwebでは「セクションヘッダー」と呼びます。
セクションヘッダーは「セクション」の始まりを示します。セクションは次のセクションヘッダーまで、あるいはファイルの終わりまで続きます。
フロントマターブロックには以下のセクション名があります:
- main
- data
- style
- meta.name
- meta.http-equiv
- meta.property
- link
ここでは、最初の3つについて説明します; 他の3つはhead要素の内容の管理 で説明します。
フロントマターブロックの「main」セクションで、「定義済みプロパティ」の値を設定します。
以下は定義済みプロパティの例です:
scheme
: HTMLドキュメントのURLのスキーム。http
またはhttps
。デフォルト:http
。host
: HTMLドキュメントのURLのホスト名。デフォルト:localhost
.port
: HTMLドキュメントのURLのポート番号。 デフォルト: 3000.url
: HTMLドキュメントのURL。root-url
: HTMLドキュメントのルートURL。title
: HTML文書のタイトル。layout
: テンプレートに適用するレイアウトの名前。Layouts を参照。html-class
:<html>
要素のclass
属性に設定された値。
通常、scheme
、host
、および port
プロパティに値を指定する必要はありません。
これらのプロパティの値は、ウェブサイトがTeamgenikで公開されるときに適切に設定されます。
url
プロパティの値は、これらのプロパティとページまたはアーティクルへのパスから生成されます。その値は読み取り専用です。
以下の3つのプロパティは、アーティクルにおいてのみ意味を持ちます:
index
: アーティクの並べ替えに使われる整数値。tags
: アーティクを分類する文字列または文字列の配列。embedded-only
:true
ならば、HTMLドキュメントは作成されない。デフォルト:false
。
詳細は アーティクル を参照してください。
shared-wrapper
プロパティはラッパーにおいてのみ意味を持ちます。
詳細は 共有ラッパー を参照してください。
draft
プロパティはページおよびアーティクルが「下書き」であることを示します。
tgweb-dist
および tgweb-server
コマンドは、下書きのページおよびアーティクルを無視します。
また、<tg:article>
と <tg:articles>
要素は、下書きのページおよびアーティクルを Web ページに埋め込みません。
また、<tg:link>
と <tg:links>
要素は、下書きのページおよびアーティクルに対するリンクを生成しません。
ただし、--buildDrafts
オプション付きで tgweb-dist
および tgweb-server
コマンドを起動した場合は、下書きのページおよびアーティクルを通常のページおよびアーティクルとして扱います。
定義済みのプロパティの値は <tg:prop>
要素によってテンプレートに埋め込むことができます。
---
[main]
title = "Our Mission"
---
<body>
<h1 class="text-2xl font-bold">
<tg:prop name="title"></tg:prop>
</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</body>
すでに述べたように、フロントマターブロックの [data]
は「data」セクションの始まりを示します。
「data」セクション内では、カスタムプロパティを定義することができます。ウェブサイト作成者は、任意の名前のカスタムプロパティに値を設定することができます。カスタムプロパティの値は、文字列か10進数表記の数値でなければなりません。
カスタムプロパティの値は <tg:data>
要素によってテンプレートに埋め込むことができます。
---
[main]
title = "Our Mission"
[data]
message = "Hello, world!"
year = 2023
---
<body>
<h1 class="text-2xl font-bold">
<tg:prop name="title"></tg:prop>
</h1>
<div class="bg-green-300 p-4">
<p><tg:data name="message"></tg:data></p>
</div>
<footer>© Example Inc. <tg:data name="year"></tg:prop></footer>
</body>
また、${...}
記法を使えば、カスタムプロパティの値をHTML要素の属性値に埋め込むこともできます。
---
[data]
div-id = "special"
---
<body>
<div id="${div-id}">...</div>
</body>
しかし、${...}
記法は class
属性に対しては意味がありません。;
プロパティの値を class
属性に埋め込む方法については後で説明します。
「style」セクションで定義されたプロパティは、クラストークンのセットにエイリアスを与えるために使用することができます。 これを「スタイル・エイリアス」と呼びます。スタイルエイリアスの値は文字列のみです。
定義されたスタイルエイリアスを class
属性の値に埋め込むには、tg:class
属性を使います。
---
[style]
blue-square = "w-24 h-24 md:w-48 md:h-48 bg-blue-500 rounded-xl p-8 m-4"
---
<body>
<div tg:class="blue-square">
Hello, world!
</div>
</body>
上記のテンプレートに含まれるエイリアスを展開すると、以下の結果が得られます:
<body>
<div class="w-24 h-24 md:w-48 md:h-48 bg-blue-500 rounded-xl p-8 m-4">
Hello, world!
</div>
</body>
クラストークンの長いシーケンスは、3つの引用符で囲んで数行にわたって書くことができます。
---
[style]
card = """
bg-white dark:bg-slate-900 rounded-lg px-6 py-8
ring-1 ring-slate-900/5 shadow-xl
"""
---
<body>
<div tg:class="card">
Hello, world!
</div>
</body>
連続する3つのダブルクォーテーション("""
)は、複数行の文字列の開始と終了を示します。
HTML要素の class
属性に埋め込まれる前に、プロパティの値は以下のように変換されます:
- 含まれる改行文字はすべてスペースに置き換えられる。
- 連続する先頭と末尾の空白は削除される。
- 連続したスペースは1つのスペースに置き換えられる。
要素が class
属性と tg:class
属性の両方を持つ場合、両方の値を合わせたものが最終的な class
属性の値となります。
レスポンシブデザイン を実現するために、Tailwind CSS修飾子を使って以下のスタイルエイリアスを定義するとよいでしょう。
box = """
w-16 h-16 p-1 border border-1 rounded
md:w-32 md:h-32 md:p-2 md:rounded-md
lg:w-48 md:h-48 lg:p-3 lg:rounded-lg
"""
md:
や lg:
のような修飾語の繰り返しが気になる場合は、次のように {...}
表記を使うと削除できます。
box = """
w-16 h-16 p-1 border border-1 rounded
md { w-32 h-32 p-2 rounded-md }
lg { w-48 h-48 p-3 rounded-lg }
"""
{...}
表記は、arbitrary variants の活用に特に役立ちます。
次のようなスタイル・エイリアス blog-article
を作成したと仮定します:
blog-article = """
text-gray-900
[&_*]:mb-2
[&_*:last-child]:mb-0
[&_h2]:text-xl
[&_h2]:font-bold
[&_h2]:tracking-wide
[&_h2]:capitalize
[&_p]:font-serif
[&_p]:leading-5
"""
次のように使用できます:
<article tg:class="blog-article">
<h2>Title</h2>
<p>Lorem ipsum dolor sit amet.</p>
<p>Consectetur adipiscing elit.</p>
<p>Sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.</p>
</article>
そうすれば、以下のように {...}
の表記を使うことで、上記の例で [&_h2]
と [&_p]
の繰り返しをなくすことができます。
blog-article = """
text-gray-900
[&_*]:mb-2
[&_*:last-child]:mb-0
[&_h2] { text-xl font-bold tracking-wide capitalize }
[&_p] { font-serif leading-5 }
"""
ページのフロントマターで特定のプロパティに値が設定されていない場合、ラッパーやレイアウトのフロントマターやサイトプロパティを検索します。詳しくは、サイトプロパティとプロパティの継承 を参照してください。
src
ディレクトリの color_scheme.toml
を編集することで、Tailwind CSSのカスタムカラー名を定義することができます。
カスタムカラー名は「パレット」と「修飾子」の組み合わせです。
パレットは3文字のアルファベット、修飾子は1文字のアルファベットです。
パレットと修飾子は、bas-s
や neu-d
のようにマイナス記号で結ばれます。
以下は利用可能なパレットとその用途の一覧です。
- bas: ベースカラー(ウェブサイト全体の背景色)
- neu: ニュートラルカラー(グレー、ベージュ、アイボリーなどの落ち着いた色)
- pri: プライマリーカラー(ベースカラーと中間色以外で最も使用頻度の高い色)
- sec: セカンダリーカラー (ベースカラーとニュートラルカラー以外で2番目によく使われる色)
- acc: アクセントカラー (ベースカラーとニュートラルカラー以外で3番目に使用頻度の高い色)
- nav: ナビゲーションカラー (ナビゲーション・バーまたはサイドバーの背景色)
以下は、修飾子とその意味の一覧です。
- s: 標準
- b: より明るい
- d: より暗い
- c: 対照的
ここでいう「対照的」とは、標準色の背景にその色でテキストを描いたときに視認性のよい色を意味します。
src/color_scheme.toml
bas-s = "#3d4451"
bas-c = "#a0aec0"
pri-s = "#45ba9f"
sec-s = "#70365d"
定義されたカスタムカラー名は、Tailwind CSSクラスを構成する色の名前として使用できます。例えば、pri-s
と定義された色を要素の背景色として設定するには、その要素の class
属性に bg-pri-s
クラスを指定します。
また、primary
, secondary
, success
, warning
のように、daisyUI で指定された色名を使うこともできます。詳細については、daisyUIドキュメンテーションの Colors を参照してください。
現時点では、tgwebはdaisyUIのテーマの切り替えをサポートしていないことに注意してください。
tgweb では Tailwind CSS を利用してビジュアルデザインを調整することが推奨されています。独自の CSS ファイルを使用したい場合は、shared_css
ディレクトリまたは css
ディレクトリに .css
ファイルを置き、フロントマターの stylesheets
にファイル名の本体をキー、値を true
とする項目を追加してください。
次に示すのは、フロントマターの設定例です:
[stylesheets]
main = true
上記の設定は、shared_css
ディレクトリまたは css
ディレクトリにある main.css
を使用するという意味になります。
通常、shared_css
ディレクトリは ../../../shared_css
へのシンボリックリンクです。shared_css
ディレクトリと css
ディレクトリに同名のファイルが存在した場合、css
ディレクトリにあるものが優先されます。
npx tgweb-archive
コマンドを用いて作られた .zip
ファイルを TeamGenik にインポートした場合、site.toml
およびフロントマターに含まれる "stylesheets" セクションは無視されます。
すなわち、TeamGenik 上で管理・公開する Web サイトにおいては、独自の CSS ファイルを使用することはできません。
tgweb では 動的要素 を利用して画面上の要素に動きをつけることが推奨されています。独自の JavaScript ファイルを使用したい場合は、shared_js
ディレクトリまたは js
ディレクトリに .js
ファイルを置き、フロントマターの javascripts
にファイル名の本体をキー、値を true
とする項目を追加してください。
次に示すのは、フロントマターの設定例です:
[javascripts]
main = true
上記の設定は、shared_js
ディレクトリまたは js
ディレクトリにある main.js
を使用するという意味になります。
通常、shared_js
ディレクトリは ../../../shared_js
へのシンボリックリンクです。shared_js
ディレクトリと js
ディレクトリに同名のファイルが存在した場合、js
ディレクトリにあるものが優先されます。
npx tgweb-archive
コマンドを用いて作られた .zip
ファイルを TeamGenik にインポートした場合、site.toml
およびフロントマターに含まれる "scripts" セクションは無視されます。
すなわち、TeamGenik 上で管理・公開する Web サイトにおいては、独自の JavaScript ファイルを使用することはできません。
ファビコンファイルは、作業ディレクトリの下の src/icons
サブディレクトリに置かれます。
ここに置かれるべきファビコンファイルは、以下の 5 種類です:
favicon.ico
-- ICO 形式の画像ファイルicon.svg
-- SVG 形式の画像ファイル180.png
-- 幅 180px、高さ 180px の PNG 形式の画像ファイル198.png
-- 幅 198px、高さ 198px の PNG 形式の画像ファイル512.png
-- 幅 512px、高さ 512px の PNG 形式の画像ファイル
ファビコンファイルの設置は義務ではありません。
画像ファイルは、作業ディレクトリの下の src/images
サブディレクトリに置かれます。
ページに画像を埋め込むには2つの方法があります。1 つは <img>
要素を使用する方法、もう 1 つは画像を要素の背景として設定する方法です。
Teamgenikは以下の形式の画像ファイルをサポートしています:
- AVIF ('.avif')
- BMP ('.bmp')
- GIF ('.gif')
- JPEG ('.jpg', '.jpeg')
- PNG ('.png')
- WEBP ('.webp')
<img>
要素の src
属性に画像ファイルの絶対パスが含まれていれば、画像はページに埋め込まれます。
<img src="/images/smile.png" alt="Smile face">
Teamgenik が自動的に指定するので、<img>
タグに width
属性と height
属性を記述する必要はありません。
src
属性の値は画像ファイルのパスは「絶対」パスであることに注意してください。ウェブサイトがTeamgenikで公開されると、<img>
要素の src
属性の値は適切に変換されます。
また、Teamgenikでは <img>
要素で外部URLを参照することはできません。
画像コンテナのサイズに合わせて画像のサイズを変更したい場合は、Tailwind CSSが提供する以下のクラストークンを使用します:
object-cover
: コンテナを覆うように画像のサイズを変更するobject-contain
: コンテナ内に収まるように画像のサイズを変更するobject-fill
: コンテナに合わせて画像を引き伸ばすobject-scale-down
: 画像をオリジナルサイズで表示し、必要に応じてコンテナに合わせて縮小する
詳細については、Tailwind CSS ドキュメントの Object Fit を参照してください。
images/smile.png
を <div>
要素の背景画像として埋め込むには、次のように記述します:
<div class="bg-[url(../images/smile.png)]"></div>
Tailwind CSS はこの class
属性を検出すると、適切なCSSフラグメントを dist/css/tailwind.css
に書き込みます。
括弧の中は css/tailwind.css
から画像ファイルまでの相対パスであることに注意してください。
src/images/smile.png
を src/pages/foo/bar.html
に埋め込む場合でも、背景画像として埋め込む場合は、括弧内に ../../images/smile.png
ではなく ../images/smile.png
を指定します。
背景画像のレンダリングを調整したい場合は、Tailwind CSSが提供する以下のクラストークンを使用してください:
bg-center
: 背景画像を背景レイヤーの中央に配置するbg-repeat
: 背景画像を縦横に繰り返すbg-repeat-x
: 背景画像を水平に繰り返すbg-repeat-y
: 背景画像を縦に繰り返すbg-cover
: 背景レイヤーを埋めるまで背景画像を拡大縮小するbg-contain
: トリミングや引き伸ばしをせずに、背景画像を外縁まで拡大縮小する
詳細については、Tailwind CSS ドキュメントの Background Position, Background Repeat , Background Size を参照してください。
Tgweb は Lottie 形式のアニメーションファイルに対応しています。アニメーションファイルは、作業ディレクトリの下の src/animations
サブディレクトリに置かれます。
アニメーションファイルの拡張子は .json
または .lottie
です。
<tg:animation>
要素でアニメーションを Web ページ内に埋め込むことができます。その src
属性にアニメーションファイルの名前を指定すればアニメーションが再生されます。次の例は、src/animations
サブディレクトリに置かれたアニメーションファイル cat.json
を再生します。
<tg:animation src="cat.json"></tg:animation>
次の例は、src/animations/common
サブディレクトリに置かれたアニメーションファイル flag.lottie
を再生します。
<tg:animation src="common/flag.lottie"></tg:animation>
デフォルトでアニメーションは 300px の幅と 150px の高さで表示されます。幅と高さを調整するには、<tg:animation>
要素に対して width
および height
属性に整数値を指定してください。
<tg:animation src="cat.json" width="100" height="100"></tg:animation>
アニメーションの背景色等を調整するには、<tg:animation>
要素に対して class
属性を指定してください。
<tg:animation src="cat.json" class="bg-red-200"></tg:animation>
<tg:animation>
要素には他に以下の属性を設定できます:
loop
: ループ再生するかどうかを示す真偽値(デフォルト:"true"
)autoplay
: 自動再生するかどうかを示す真偽値(デフォルト:"true"
)click
: クリックまたはタップで再生状態をトグルするかどうかを示す真偽値(デフォルト:"false"
)hover
: マウスホーバー中のみ再生するかどうかを示す真偽値(デフォルト:"false"
)
click
属性と hover
属性の値がともに "true"
の場合、click
属性のみが有効となります。
また、autoplay
属性が "true"
の場合、hover
属性は無視されます。
アニメーションファイルの入手または作成する方法については、LottieFiles を参照してください。
音声ファイルは、作業ディレクトリの下の src/audio
サブディレクトリに置かれます。
Teamgenikは以下の形式の音声ファイルをサポートしています:
- AAC ('.m4a')
- MP3 ('.mp3')
- Ogg Vorbis (
.ogg
) - WAV (
.wav
)
<audio>
要素で音声コンテンツを再生するUIオブジェクトを埋め込むことができます。
<audio>
要素を構成するには2つの方法があります。
ひとつは、<audio>
要素自体の src
属性に音声ファイルの「絶対」パスを指定する方法です。
<audio controls src="/audios/theme.mp3">
<a href="/audios/theme.mp3">Download</a>
</audio>
ブラウザが <audio>
要素をサポートしていない場合、<audio>
要素の内容が表示されます。
もう一つは、<audio>
要素の中に1つ以上の <source>
要素を配置し、その src
属性に音声ファイルの絶対パスを指定する方法です。
<audio controls>
<source src="/audios/theme.ogg" type="audio/ogg">
<source src="/audios/theme.mp3" type="audio/mpeg">
Your browser does not support the audio element.
</audio>
Teamgenikでは、<audio>
要素と <source>
要素で外部URLを参照することはできません。
Google Fonts のRobotoフォントファミリーは、site.toml
の "font.google-fonts" セクションで以下のように設定することで、あなたのウェブサイトで使用することができます。
Roboto = true
以下はRobotフォントファミリーの使用例です:
<p class="font-['Roboto']">Hello, world!</p>
フォントファミリ名にスペースが含まれている場合は、下記のように二重引用符で囲む必要があります:
"Noto Sans JP" = true
また、HTMLテンプレートで使用する場合は、スペースをアンダースコアに置き換えてください。
<p class="font-['Noto_Sans_JP']">こんにちは、世界!</p>
フォントファイルのサイズを小さくするために、いくつかのフォントウェイト(太さ)を選択するには、true
の代わりにウェイトを配列で指定します。
"Noto Sans JP" = [400, 800]
次の例では、ウェイトが 800 の Noto Sans JP フォントファミリーを使用しています。
<p class="font-['Noto_Sans_JP'] font-[800]">こんにちは、世界!</p>
各スタイルのフォントのウェイトを選択するには、以下のようにインライン・テーブルを使ってウェイトを指定します:
"Pathway Extreme" = { normal = [400, 800], italic = [400] }
以下の例では、イタリック体の Pathway Extreme フォントファミリーをウェイトを 400 で使用しています。
<p class="font-['Pathway_Extreme'] italic font-[400]">Hello, world!</p>
tgweb では、Google が提供する マテリアルシンボル が利用可能です。マテリアルシンボルはアイコンフォントの一種です。この文書ではマテリアル シンボルを単に「シンボル」と呼びます。
初期状態の site.toml
には次のような記述が含まれています。
[font.material-symbols]
outlined = true
rounded = true
sharp = true
このように書かれている場合、3 種類の「スタイル」、すなわち「outlined」、「rounded」、「sharp」のシンボルがすべて利用可能です。あなたのウェブサイトで一部または全部のスタイルを利用しない場合は、sites.toml
の font.material-symbols
セクションの outlined
、rounded
、sharp
プロパティの値を false
に設定し、ウェブサイト訪問者の負荷を軽減することおすすめします
例えば、次のように書けば、「sharp」スタイルのシンボルのみが読み込まれます。
[font.material-symbols]
outlined = false
rounded = false
sharp = true
シンボルには書体を調整するための4つの「変数」が用意されています。これらを指定するには、次のように記述してください:
[font.material-symbols]
rounded = { fill = 1, wght = 200, grad = 0, opsz = 24 }
sharp = { fill = 0, wght = 300, grad = 200, opsz = 40 }
fill
は塗りつぶしの有無を制御する変数です。0 なら「なし」、1 なら「あり」を意味します。デフォルト値は0です。
wght
はシンボルの「太さ(weight)」を決定する変数です。100, 200, 300, 400, 500, 600, 700 のいずれかを指定できます。100が最も細く、700が最も太くなります。デフォルト値は400です。
grad
はシンボルの「グレード(grade)」を決定する変数です。この変数の値を変更することで、シンボルの太さを微調整できます。指定できる値は -25, 0, 200 のいずれかです。デフォルト値は0です。負の値を指定するとシンボルの線がより細くなり、正の値を指定するとシンボルの線がより太くなります。
opsz
はシンボルの「光学サイズ(optical size)」を決定する変数です。光学サイズとは、シンボルの推奨表示サイズを示します。20, 24, 40, 48 のいずれかを指定できます。デフォルト値は24です。一般に、光学サイズの値を大きくすると、線がより細く、空間がより狭く、x-height (ベースラインと書体の小文字の平均線との間の距離)がより短くなります。
ひとつのスタイルに対して変数の値が異なる複数のバリアントを用意したい場合は、次のようにスタイル名の後にマイナス記号とバリアント名を加えてください。
[font.material-symbols]
outlined = true
sharp = true
rounded = { fill = 0, wght = 200, grad = 0, opsz = 24 }
rounded-strong = { fill = 1, wght = 400, grad = 0, opsz = 24 }
rounded-bold = { fill = 0, wght = 700, grad = 0, opsz = 24 }
rounded-very-thin = { fill = 0, wght = 100, grad = 0, opsz = 24 }
バリアント名には小文字 (a-z
) と数字 (0-9
) とマイナス記号(-
)しか使えないことに注意してください。
ページ、アーティクル、セグメント等にシンボルを埋め込むには、配置したい場所に <tg:symbol>
要素を追加し、name
属性でシンボル名、symbol-style
属性でシンボルスタイルを指定します。symbol-style
属性が省略された場合、"outlined"
が指定されたものとみなされます。
例えば、次の例は「outlined」スタイルの「Home」シンボルを生成します。
<tg:symbol name="home" symbol-style="outlined"></tg:symbol>
シンボルの名前は「スネークケース」に変換する必要があります。つまり、名前に含まれるすべてのスペースをアンダースコアに、すべての大文字を小文字に置き換えてください。例えば、「Shopping Bag」シンボルを生成するには、name
属性に "shopping_bag"
を指定してください。
<tg:symbol name="shopping_bag" symbol-style="sharp"></tg:symbol>
変数を指定してシンボルの書体を調整するには、fill
属性、wght
属性、grad
属性、opsz
属性を指定してください。
<tg:symbol name="star" symbol-style="rounded" fill="1"></tg:symbol>
フォントのサイズを指定するには、Tailwind の text-xs
や text-lg
などのクラスを利用してください。
<tg:symbol name="favorite" symbol-style="rounded" fill="1" class="text-lg"></tg:symbol>
通常、ウェブサイトのページには、ほとんどのコンテンツを共有する一連の領域があります: ヘッダー、サイドバー、フッターなどです。
この一連の領域をひとつのテンプレートとしてページから分離すると、ウェブサイトの管理が容易になります。この分割されたテンプレートを「レイアウト」と呼びます。
レイアウトは、作業ディレクトリ下の src/layouts
サブディレクトリに置かれるHTMLファイルです。
レイアウトは次の 3 つの条件を満たす必要があります:
- トップレベル要素は1つしかない。
- トップレベル要素は
<body>
要素である。 - トップレベル要素は、その子孫要素内に
<tg:content>
要素を1つだけ含む。
<tg:content>
要素は、レイアウトのどこにページが挿入されるかを示します。
src/layouts/common.html
<body>
<header>
<div>Example</div>
</header>
<main>
<tg:content></tg:content>
</main>
<footer>© Example Inc. 2023</footer>
</body>
<tg:content></tg:content>
の代わりに <tg:content />
を書くことは「できない」ことに注意してください。
このレイアウトをページに適用するには、ページのフロントマターの layout
プロパティにレイアウト名を指定します。
レイアウトの名前は、レイアウトのファイル名から拡張子(.html
)を除いたものです。この場合、common
がレイアウトの名前です。
src/pages/index.html
---
[main]
layout = "common"
---
<h1>Welcome!</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
前の例で示したレイアウト common
をこのページファイルに適用すると、次のようなHTML文書が生成されます:
<html>
<head>
...
</head>
<body>
<header>
<div>Example</div>
</header>
<main>
<h1>Welcome!</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</main>
<footer>© Example Inc. 2023</footer>
</body>
</html>
ページのフロントマターで設定されたプロパティの値は、<tg:prop>
要素を使ってレイアウトに埋め込むことができます。
src/layouts/common.html
<body>
<header>
<div>Example</div>
</header>
<main>
<h1 class="text-3xl"><tg:prop name="title"></tg:prop></h1>
<tg:content></tg:content>
</main>
</body>
src/pages/greeting.html
---
[main]
layout = "common"
title = "Greeting"
---
<h1>Welcome!</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
<tg:slot>
要素はレイアウト内のプレースホルダーで、ページ内で指定した内容を埋めることができます。<tg:slot>
要素には name
属性を指定する必要があります。
レイアウトのスロットにコンテンツを埋め込むには、レイアウトを適用するページに <tg:insert>
要素を配置します。
<tg:insert>
要素の name
プロパティの値としてスロット名を指定すると、スロットは<tg:insert>
要素の内容に置き換えられます。
ページコンテンツがレイアウト内の <tg:content>
要素に挿入されると、すべての <tg:insert>
要素がページコンテンツから削除されます。
src/layouts/product.html
<body class="p-2">
<tg:content></tg:content>
<div class="border-2 border-black border-solid p-2">
<tg:slot name="remarks"></tg:slot>
</div>
<div><tg:slot name="badges"></tg:slot></div>
</body>
src/pages/product1.html
---
[main]
layout = "product"
title = "Product 1"
---
<div>
<h1>Product 1</h1>
<p>Description</p>
</div>
<tg:insert name="remarks">This product is very <em>fragile</em>.</tg:insert>
<tg:insert name="badges"><span>A</span><span>B</span></tg:insert>
レイアウト product
がページ product1.html
に適用されると、次のようなHTML文書が生成される:
<html>
<head>
...
</head>
<body class="p-2">
<div>
<h1>Product 1</h1>
<p>Description</p>
</div>
<div class="border-2 border-black border-solid p-2">
This product is very <em>fragile</em>.
</div>
<div><span>A</span><span>B</span></div>
</body>
</html>
スロットに挿入されるコンテンツが定義されていない場合、<tg:slot>
要素の内容が代替コンテンツとして使用されます。
src/layouts/message.html
<body class="p-2">
<tg:content></tg:content>
<div><tg:slot name="message">No message.</tg:slot></div>
</body>
src/pages/home.html
---
[main]
layout = "message"
title = "Home"
---
<h1>Home</h1>
<p>Hello, world!</p>
レイアウト message
がページ home.html
に適用されると、次のようなHTML文書が生成されます:
<html>
<head>
...
</head>
<body class="p-2">
<h1>Home</h1>
<p>Hello, world!</p>
<div>No message.</div>
</body>
</html>
通常、レイアウト内の <tg:if-complete>
要素は単にその内容で置き換えられます。
ただし、以下の3つの条件を満たさない場合は、要素全体が削除されます:
- その中のすべての
<tg:prop>
要素に挿入されるプロパティ値が定義されている。 - その中のすべての
<tg:data>
要素に挿入されるカスタムプロパティ値が定義されている。 - その中のすべての
<tg:slot>
要素に挿入される内容が定義されている。
src/layouts/message.html
<body class="p-2">
<tg:content></tg:content>
<tg:if-complete>
<hr class="h-px my-8 bg-gray-200 border-0">
<div class="bg-gray-800 text-white p-4">To: <tg:data name="custom-name"></tg:data></div>
<div class="bg-gray-200 p-4"><tg:slot name="message"></tg:slot></div>
</tg:if-complete>
</body>
src/pages/home1.html
---
[main]
layout = "message"
[data]
custom-name = "Alice"
---
<h1>Home</h1>
<tg:insert name="message">Hello, world!</tg:insert>
レイアウト message
がページ home1.html
に適用されると、次のようなHTML文書が生成されます:
<html>
<head>
...
</head>
<body class="p-2">
<h1>Home</h1>
<hr class="h-px my-8 bg-gray-200 border-0">
<div class="bg-gray-800 text-white p-4">To: Alice</div>
<div class="bg-gray-200 p-4">Hello, world!</div>
</body>
</html>
src/pages/home2.html
---
[main]
layout = "message"
[data]
custom-name = "Alice"
---
<h1>Home</h1>
レイアウト message
がページ home2.html
に適用されると、次のようなHTML文書が生成されます:
<html>
<head>
...
</head>
<body class="p-2">
<h1>Home</h1>
</body>
</html>
ラッパーとは、レイアウトとページの中間のレベルに存在するテンプレートのことです。
ラッパーを使えば、共通のスタイルや共通の要素を一群のページに追加できます。
ラッパーのファイル名は常に _wrapper.html
です。作業ディレクトリ下のsrc/pages
サブディレクトリの直下かその子孫ディレクトリに配置します。
あるディレクトリーに置かれたラッパーは、そのディレクトリーと子孫ディレクトリー内のすべてのページに適用されます。
もし、あるディレクトリにラッパーが存在しない場合、ディレクトリ階層で下から順番に最初に見つかったラッパーが、そのディレクトリのラッパーとなります。
例えば、src/pages/foo/_wrapper.html
が存在し、_wrapper.html
が src/pages/foo/bar
ディレクトリにも src/pages/foo/bar/baz
ディレクトリにも存在しない場合、src/pages/foo/_wrapper.html
は src/pages/foo/bar/baz
ディレクトリのラッパーとなります。
基本的に、ラッパーはレイアウトと同じように書きます。
<tg:content>
要素は、ラッパー内のどこにページが挿入されるかを示します。
src/pages/mission/_wrapper.html
<h1 class="text-xl bg-blue-400 p-2">Our Mission</h1>
<div class="[&_p]:mt-2 [&_p]:pl-2">
<tg:content></tg:content>
</div>
class
属性値 [&_p]:mt-2 [&_p]:pl-2
は、このラッパー内の全ての <p>
要素の margin-top
と padding-left
をスケール2(8px/0.5rem)に設定します。[&_p]
の表記については Using arbitrary variantsを参照してください。
すでに述べたように、あるディレクトリーに置かれたラッパーは、そのディレクトリーと子孫ディレクトリー内のすべてのページに適用されます。この例では、src/pages/mission/
ディレクトリとその子孫ディレクトリ src/pages/mission/member_mission/
に次の2つのページが存在するとします。
src/pages/mission/team_mission.html
<h2 class="text-lg">[Team]</h2>
<div>
<p>Create workshop pages.</p>
</div>
src/pages/mission/member_mission/alice_mission.html
<h2 class="text-lg">[Alice]</h2>
<div>
<p>Coding with html (workshop page).</p>
<p>Coding with html (workshop calender page).</p>
</div>
ラッパー _wrapper.html
がこれら2つのページに適用されるので、それぞれの body
部は次のように生成されます。
dist/pages/mission/team_mission.html
<body>
<h1 class="text-xl bg-blue-400 p-2">Our Mission</h1>
<div class="[&_p]:mt-2 [&_p]:pl-2">
<h2 class="text-lg">[Team]</h2>
<div>
<p>Create workshop pages.</p>
</div>
</div>
</body>
dist/pages/mission/member_mission/alice_mission.html
<body>
<h1 class="text-xl bg-blue-400 p-2">Our Mission</h1>
<div class="[&_p]:mt-2 [&_p]:pl-2">
<h2 class="text-lg">[Alice]</h2>
<div>
<p>Coding with html (workshop page).</p>
<p>Coding with html (workshop calender page).</p>
</div>
</div>
</body>
どちらのページも、<p>
要素の margin-top
と padding-left
はスケール2(8px/0.5rem)に設定されます。このように特定のディレクトリーとその子孫ディレクトリー内のすべてのページに共通のスタイルや共通の要素を追加できます。
ラッパーのフロントマターで設定されたプロパティの値は、そのラッパーが適用されるページのプロパティのデフォルト値になります。もし、ページのフロントマターで同じプロパティに値が設定されている場合は、ページでの設定が優先されます。
ラッパーのプロパティ値は、サイトのプロパティ値よりも優先されます。
この例では、ラッパーのプロパティ値として title = "Team's Mission"
が定義されています。そして、<tg:prop>
要素を使って h1
要素の内容として記述されています。
src/pages/mission/_wrapper.html
---
[main]
title = "Team's Mission"
---
<h1 class="text-xl bg-blue-400 p-2">
<tg:prop name="title"></tg:prop>
</h1>
<div class="[&_p]:mt-2 [&_p]:pl-2">
<tg:content></tg:content>
</div>
上記のラッパーを次の2つのページに適用します。
src/pages/mission/team_mission.html
<div>
<p>Create workshop pages.</p>
</div>
src/pages/member_mission/alice_mission.html
---
[main]
title = "Alice's Mission"
---
<div>
<p>Coding with html (workshop page).</p>
<p>Coding with html (workshop calender page).</p>
</div>
team_mission.html
の方は、フロントマターでプロパティ title
を再定義していませんが、member_mission/alice_mission.html
の方は再定義しています。
ラッパー _wrapper.html
がこれら2つのページに適用されると、それぞれの body
部は次のように生成されます。
dist/pages/mission/team_mission.html
<body>
<h1 class="text-xl bg-blue-400 p-2">
Team's Mission
</h1>
<div class="[&_p]:mt-2 [&_p]:pl-2">
<div>
<p>Create workshop pages.</p>
</div>
</div>
</body>
dist/pages/mission/member_mission/alice_mission.html
<body>
<h1 class="text-xl bg-blue-400 p-2">
Alice's Mission
</h1>
<div class="[&_p]:mt-2 [&_p]:pl-2">
<div>
<p>Coding with html (workshop page).</p>
<p>Coding with html (workshop calender page).</p>
</div>
</div>
</body>
team_mission.html
の方は、ラッパーで定義したのプロパティ title
がレンダリングされています。一方、member_mission/alice_mission.html
の方はページで再定義した title
がレンダリングされています。
複数のサイト間で共有されるラッパーを共有ラッパー (shared wrappers)と呼びます。
共有ラッパーは、作業ディレクトリの src/shared_wrappers
サブディレクトリに置かれます。通常、このサブディレクトリは他のディレクトリへのシンボリックリンクです。
複数のウェブサイトを管理する方法で説明した方法でウェブサイトの作業ディレクトリを初期化した場合、src/shared_wrappers
サブディレクトリは ../../../shared_wrappers
ディレクトリを参照するシンボリックリンクとして作られます。
"decorated.html"
という名前の共通ラッパーをあるディレクトリのラッパーとして使用したければ、
そのディレクトリに次のような中身を持つ _wrapper.html
を作成してください:
---
[main]
shared-wrapper = "decorated"
---
「segment」はページ、レイアウト、セグメントに埋め込むことができるテンプレートファイルです。セグメントはアーティクル、ラッパー、部品など、これらのタイプ以外のテンプレートに埋め込むことはできません。
セグメントを別のセグメントに埋め込むときは、循環参照を避けるように注意する必要があります。循環参照が検出された場合、生成されたHTMLにエラーメッセージが挿入されます。
セグメントは作業ディレクトリの src/segments
サブディレクトリに置かれます。
以下はセグメントの例です :
src/segments/hero.html
<section class="bg-base-200 p-4 border border-rounded">
<div class="flex">
<img src="/images/hello.jpg">
<div>
<h1 class="text-5xl font-bold">Hello, world!</h1>
<p>The quick brown fox jumps over the lazy dog.</p>
</div>
</div>
</section>
ページにセグメントを埋め込むには、配置したい場所に <tg:segment>
要素を追加し、name
属性でセグメント名を指定します。
<div>
<tg:segment name="hero"></tg:segment>
<main>
...
</main>
</div>
<tg:segment>
要素の data-*
属性を使って、セグメントにカスタムプロパティを渡すことができます。
src/segment/hero.html
<section class="bg-base-200 p-4 border border-rounded">
<div class="flex">
<img src="/images/${image-path}">
</div>
</section>
src/pages/index.html
<div>
<tg:segment name="hero" data-image-path="hello.jpg"></tg:segment>
<main>
...
</main>
</div>
レイアウトと同様、スロットをセグメント内に配置できます。セグメント内のスロットにコンテンツを埋め込む方法は、レイアウトと同様です。
src/segment/hero.html
<section class="bg-base-200 p-4 border border-rounded">
<div class="flex">
<img src="/images/hello.jpg">
<div>
<h1 class="text-5xl font-bold">Hello, world!</h1>
<tg:slot name="paragraph"></tg:slot>
</div>
</div>
</section>
src/pages/index.html
<div>
<tg:segment name="hero">
<tg:insert name="paragraph">
<p>The quick brown fox jumps over the lazy dog.</p>
</tg:insert>
</tg:segment>
<main>
...
</main>
</div>
<tg:prop>
要素と <tg:data>
要素は、セグメントにプロパティの値を埋め込むことができます。
---
[data]
message = "Hi!"
---
<div>
<tg:data name="message"></tg:data>
</div>
セグメントは、それが埋め込まれているページやレイアウトからプロパティを継承することに注意してください。同じ名前のプロパティが部品とページまたはレイアウトで定義されている場合、ページまたはレイアウトで定義されている値が優先されます。
「部品」はページ、セグメント、部品、アーティクル、レイアウトに埋め込むことができるテンプレートファイルです。
部品は、作業ディレクトリの src/components
サブディレクトリに置かれます。
以下は部品の例です:
src/components/smile.html
<span class="inline-block border-solid border-2 border-black rounded p-2">
<tg:symbol name="sentiment_satisfied"></tg:symbol>
</span>
上記のシンボルを表示するには、sites.toml
の font.material-symbols
セクション内のプロパティ(この例では outlined
プロパティ)を true
に設定する必要があることに注意してください。詳しくは シンボルを参照してください。
部品を別の部品に埋め込む場合は、循環参照を避けるように注意する必要があります。循環参照が検出された場合、生成されたHTMLにエラーメッセージが挿入されます。
部品をページやアーティクル、レイアウトに埋め込むには、配置したい場所に <tg:component>
要素を追加し、name
属性でその名前を指定します。
<p>
<tg:component name="smile"></tg:component>
How are you?
</p>
<tg:component>
要素の data-*
属性を使って、カスタムプロパティを部品に渡すことができます。
src/components/avatar.html
<div class="bg-gray-200 p-2">
<a href="articles/members/${name}.html">
<img class="w-24 h-24 rounded-full" src="/images/${name}.png" alt="${name}">
</a>
</div>
src/pages/index.html
<div class="grid grid-rows-2 gap-4">
<tg:component name="avatar" data-name="Alice"></tg:component>
<tg:component name="avatar" data-name="Bob"></tg:component>
<tg:component name="avatar" data-name="Carol"></tg:component>
</div>
レイアウトやセグメントと同様、スロットを部品内に配置できます。部品内のスロットにコンテンツを埋め込む方法は、レイアウトやセグメントと同様です。
src/components/blog_item.html
<h3 class="font-bold text-lg m-2">
<tg:slot name="title"></tg:slot>
</h3>
<div>
<tg:slot name="body"></tg:slot>
</div>
<tg:if-complete>
<div class="text-right">
<tg:slot name="date"></tg:slot>
</div>
</tg:if-complete>
src/pages/hello.html
<main class="bg-gray-100 py-2">
<tg:component name="blog_item">
<tg:insert name="title">Greeting</tg:insert>
<tg:insert name="body">
<p>Hello.</p>
</tg:insert>
<tg:insert name="date">2022-12-31</tg:insert>
</tg:component>
</main>
<tg:prop>
要素と <tg:data>
要素は、部品にプロパティの値を埋め込むことができます。
---
[data]
message = "Hi!"
---
<span>
<span class="material-symbols-outlined">sentiment_satisfied</span>
<tg:data name="message"></tg:data>
</span>
部品は、それが埋め込まれているページやアーティクルからプロパティを継承することに注意してください。同じ名前のプロパティが部品とページまたはアーティクルで定義されている場合、ページまたはアーティクルで定義されている値が優先されます。
複数のサイト間で共有される部品を共有部品 (shared components)と呼びます。
共有部品は、作業ディレクトリの src/shared_components
サブディレクトリに置かれます。通常、このサブディレクトリは他のディレクトリへのシンボリックリンクです。
複数のウェブサイトを管理する方法で説明した方法でウェブサイトの作業ディレクトリを初期化した場合、src/shared_components
サブディレクトリは ../../../shared_components
ディレクトリを参照するシンボリックリンクとして作られます。
共有部品の作り方は部品と同じです。共有部品でもスロットを利用できます。
共有部品をページやアーティクル、レイアウト、部品に埋め込むには、配置したい場所に <tg:shared-component>
要素を追加し、name
属性でその名前を指定します。
共有部品の中に別の共有部品を埋め込むことは可能ですが、共有部品の中に普通の部品を埋め込むことはできません。
<p>
<tg:shared-component name="smile"></tg:shared-component>
How are you?
</p>
ページと同様、「アーティクル」はHTML文書を生成するためのテンプレート・ファイルです。
アーティクルは、作業ディレクトリの src/articles
サブディレクトリに置かれます。src/articles
ディレクトリの下にサブディレクトリを作り、その下にアーティクルを置くことも可能です。
アーティクルは完全なHTML文書に変換され、dist/articles
ディレクトリに書き込まれます。
例えば、src/articles/tech.html
は dist/articles/tech.html
に、src/articles/blog/happy_new_year.html
は dist/articles/blog/happy_new_year.html
に変換されます。
アーティクルとページは、以下の点でまったく同じ特徴を持っています:
- レイアウトを適用することができる。
- 画像や音声を埋め込むこともできる。
<head>
要素の内容は、後述 する方法で自動的に生成される。.
ここでは、アーティクルを作成し、それをブラウザで表示する方法を説明します。
次の内容で src/articles/glossary.html
を作成します。
---
[main]
title = "Glossary"
---
<h3>Glossary</h3>
<p>HTML: HyperText Markup Language</p>
<p>CSS: Cascading Style Sheets</p>
次のような <html>
や <head>
要素を含む完全なHTML文書として、dist/articles/glossary.html
が生成されます。
<!DOCTYPE html>
<html>
<head>
...
</head>
<body>
<h1>Glossary</h1>
<p>HTML: HyperText Markup Language</p>
<p>CSS: Cascading Style Sheets</p>
</body>
</html>
URLとして http://localhost:3000/articles/glossary.html
を指定するとブラウザで表示することができます。
部品のように、アーティクルはページやセグメントに埋め込むことができます。ページのラッパーにも埋め込むことができます。
<tg:article>
要素を次のようにアーティクルを埋め込みたい場所に配置します:
---
[main]
layout = "home"
---
<main>
<h1>Our Recent Articles</h1>
<tg:article name="blog/got_a_gadget"></tg:article>
<tg:article name="blog/happy_new_year"></tg:article>
</main>
<tg:article>
要素の name
属性の値は、拡張子 (.html
) を除いたアーティクルファイル名にする必要があります。
部品とは異なり、アーティクルはページにのみ埋め込むことができます。アーティクルを他のアーティクルやレイアウトに埋め込むことはできません。
レイアウトやセグメントと同様に、スロットはアーティクルの中に置くことができます。また、 <tg:article>
要素の data-*
属性を使ってアーティクルにカスタムプロパティを渡すこともできます。
アーティクルの embedded-only
プロパティの値が true
に設定されている場合、完全なHTMLファイルには変換されず、ページやセグメントへの埋め込みにのみ使用されます。
---
[main]
embedded-only = true
---
<h3>Greeting</h3>
<p>Hello, world!</p>
<tg:if-embedded>
要素はアーティクルのテンプレートの中で使われ、アーティクルが他のページやセグメントの中に埋め込まれた時にのみ表示されます。
<tg:unless-embedded>
要素はアーティクルのテンプレートの中で使われ、アーティクルが個別のウェブページとして生成される時にのみ表示されます。
<tg:articles>
要素は、複数のアーティクルをページやセグメントに埋め込むために使うことができます。
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<tg:articles pattern="proposals/*"></tg:articles>
</main>
上記の例では、src/articles/proposals
ディレクトリの直下にあるすべてのアーティクルを <h1>
要素の下に埋め込んでいます。
src/articles/proposals
ディレクトリよりも下にあるすべてのアーティクルを再帰的に埋め込むには pattern
属性に指定する値を proposals/**
と書き換えてください。
デフォルトでは、アーティクルはファイル名の昇順(アルファベット順)でソートされます。
アーティクルをファイル名の降順に並べ替えるには、<tg:articles>
要素の order-by
属性を "filename:desc"
に設定します:
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<tg:articles pattern="proposals/*" order-by="filename:desc"></tg:articles>
</main>
<tg:articles>
要素の data-*
属性を使ってアーティクルにカスタムプロパティを渡すことができます。
アーティクルをタイトルでソートするには、<tg:articles>
要素の order-by
属性を "title:asc"
または "title:desc"
に設定します:
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<tg:articles pattern="proposals/*" order-by="title:asc"></tg:articles>
</main>
アーティクルを作成日時で並べ替えるには、各アーティクルの created-at
プロパティにローカル日時を設定します:
---
[main]
created-at = 2025-01-01T09:30:00
---
<article>
...
</article>
ローカル日時は 2025-01-01T09:30:00
のように、日付と時刻を T
で連結する必要があります。
そして、<tg:articles>
要素の order-by
属性を "created-at:asc"
または "created-at:desc"
に設定します:
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<tg:articles pattern="proposals/*" order-by="created-at:asc"></tg:articles>
</main>
アーティクルを更新日時で並べ替えるには、各アーティクルの updated-at
プロパティにローカル日時を設定します:
---
[main]
updated-at = 2025-01-01T09:30:00
---
<article>
...
</article>
ローカル日時は 2025-01-01T09:30:00
のように、日付と時刻を T
で連結する必要があります。
そして、<tg:articles>
要素の order-by
属性を "updated-at:asc"
または "updated-at:desc"
に設定します:
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<tg:articles pattern="proposals/*" order-by="updated-at:asc"></tg:articles>
</main>
アーティクルを任意の順序で並べ替えるには、各アーティクルの index
プロパティを整数値に設定します:
---
[main]
index = 123
---
<article>
...
</article>
そして、<tg:articles>
要素の order-by
属性に "index:asc"
または "index:desc"
を設定します:
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<tg:articles pattern="proposals/*" order-by="index:asc"></tg:articles>
</main>
index
プロパティを持たないアーティクルは、index
プロパティを持つアーティクルの後にファイル名の昇順で並べられます。
order-by
属性の値は、コロン一つで区切られた文字列です。コロンの左側は "title"
、"index"
、"filename"
のいずれかです。コロンの右側は "asc"
または "desc"
でなければならず、"asc"
は「昇順」、"desc"
は「降順」を意味します。
テンプレートには、ページ、レイアウト、ラッパー、セグメント、部品、アーティクルがあります。この中のセグメント、部品、アーティクルは、それぞれ <tg:component>
、<tg:segment>
、<tg:article>
、<tg:articles>
要素で他のテンプレートに埋め込むことができます。
ここで、何が何に埋め込めるのかを整理しておきます。
埋め込み先↓ | <tg:component> |
<tg:segment> |
<tg:article> |
<tg:articles> |
---|---|---|---|---|
ページ | ◯ | ◯ | ◯ | ◯ |
ページのラッパー | ◯ | ◯ | ◯ | ◯ |
レイアウト | ◯ | ◯ | ✕ | ✕ |
セグメント | ◯ | ◯ | ◯ | ◯ |
部品 | ◯ | ✕ | ✕ | ✕ |
アーティクル | ◯ | ✕ | ✕ | ✕ |
アーティクルのラッパー | ◯ | ✕ | ✕ | ✕ |
<a>
要素を使ってウェブサイト内のページにリンクする場合は、href
属性でページの「絶対」パスを指定します:
<nav>
<a href="/articles/goal.html">Our Goal</a>
<a href="/articles/about.html">About Us</a>
</nav>
ウェブサイトが Teamgenik で公開されると、その中の <a>
要素の href
属性の値は適切に変換されます。
<tg:link>
は、<a>
要素を条件付きで出現させるために使われる特別な要素です。
基本的に、この要素の内容はそのままレンダリングされます。
<tg:link>
要素の中に "#"
の href
属性を持つ <a>
要素がある場合、<tg:link>
要素の href
属性の値が <a>
要素の href
属性に設定されます。
次のコードは <a href="/articles/goal">Our Goal</a>
としてレンダリングされます:
<tg:link href="/articles/goal.html">
<a href="#">Our Goal</a>
</tg:link>
しかし、/articles/goal.html
というパスを持つアーティクルが上記のコードを含む場合、この部分は生成されたHTMLドキュメントから削除されます。
0個または1個の <tg:if-current>
要素を <tg:link>
要素の中に入れることができます。
<tg:if-current>
要素の内容は、<tg:link>
要素の href
属性の値が、生成されるHTML文書のパスと一致する場合にのみレンダリングされます。
次のコードは、/articles/goal.html
というパスを持つアーティクルの中で <span class="font-bold">Our Goal</span>
としてレンダリングされます。
<tg:link href="/articles/goal.html">
<a href="#">Our Goal</a>
<tg:if-current>
<span class="font-bold">Our Goal</span>
</tg:if-current>
</tg:link>
<tg:label>
要素を使えば、上記のコードから重複を取り除くことができます。
<tg:link href="/articles/goal.html" label="Our Goal">
<a href="#"><tg:label></tg:label></a>
<tg:if-current>
<span class="font-bold"><tg:label></tg:label></span>
</tg:if-current>
</tg:link>
この要素は、<tg:link>
要素の label
属性で指定された値で置き換えられます。
src/components/nav.html
<nav>
<tg:link href="/articles/goal.html" label="Our Goal">
<a href="#" class="underline text-blue-500"><tg:label></tg:label></a>
<tg:if-current>
<span class="font-bold"><tg:label></tg:label></span>
</tg:if-current>
</tg:link>
<tg:link href="/articles/about.html" label="About Us">
<a href="#" class="underline text-blue-500"><tg:label></tg:label></a>
<tg:if-current>
<span class="font-bold"><tg:label></tg:label></span>
</tg:if-current>
</tg:link>
</nav>
<tg:link>
要素に component
属性があると、その値に対応する名前の部品の内容が <tg:link>
要素の内容になります。
例えば、次のような内容の nav_link
部品があるとします。
src/components/nav_link.html
<a href="#" class="underline text-blue-500"><tg:label></tg:label></a>
<tg:if-current>
<span class="font-bold"><tg:label></tg:label></span>
</tg:if-current>
この場合、<tg:link>
要素は次のように構成できます:
<tg:link component="nav_link" href="/articles/goal.html" label="Our Goal"></tg:link>
上記のコードは、以下のコードとまったく同じと解釈されます。
<tg:link href="/articles/about.html" label="About Us">
<a href="#" class="underline text-blue-500"><tg:label></tg:label></a>
<tg:if-current>
<span class="font-bold"><tg:label></tg:label></span>
</tg:if-current>
</tg:link>
<tg:links>
要素は、あらゆるテンプレート(ページ、レイアウト、部品、アーティクル)にアーティクルへのリンクを埋め込むために使用できます。
---
[main]
layout = "home"
---
<main>
<h1>Our Proposals</h1>
<ul>
<tg:links pattern="proposals/*">
<li>
<a href="#">
<tg:prop name="title"></tg:prop>
<tg:if-complete>
<span class="text-sm">
(<tg:slot name="date"></tg:slot>)
</span>
</tg:if-complete>
</a>
</li>
</tg:links>
</ul>
</main>
<tg:links>
要素は1つ以上の <a>
要素と0つ以上の <tg:prop>
要素、<tg:data>
要素、<tg:slot>
要素を含みます。
<a>
の href
属性の値は、アーティクルのURLに置き換えられます。<tg:prop>
要素と <tg:data>
要素は、埋め込まれるアーティクルのプロパティの値で置き換えられます。<tg:slot>
要素は、埋め込まれるアーティクル内で定義された <tg:insert>
要素の内容に置き換えられます。
<tg:links>
要素の内側では、<tg:if-complete>
要素は部品の内側と同じように機能します。つまり、<tg:if-complete>
要素は、その中のすべての <tg:prop>
、<tg:data>
、<tg:slot>
要素に値または内容が提供されない限り、出力から削除されます。
デフォルトでは、アーティクルはファイル名の昇順(アルファベット順)でソートされます。アーティクルをタイトルでソートするには、<tg:links>
要素の order-by
属性を "title:asc"
または "title:desc"
に設定します:
<tg:links pattern="proposals/*" order-by="title:asc">
<li>
...
</li>
</tg:links>
order-by
属性に設定する値の書き方については、アーティクルをタイトルで並べ替えるを参照してください。
アーティクルを任意の順序で並べ替えるには、各アーティクルのフロントマターに index
プロパティを追加します:
---
[main]
layout = "home"
index = 123
---
<article>
...
</article>
次に、<tg:links>
要素の order-by
属性を指定します。
<tg:links pattern="proposals/*" order-by="index:asc">
<li>
...
</li>
</tg:links>
現在の仕様では、<tg:component>
要素を <tg:links>
要素の中に入れることはできません。
0個または1個の <tg:if-current>
要素を <tg:links>
要素の中に入れることができます。<tg:if-current>
要素の内容は、アーティクルのパスが生成されるHTMLドキュメントのパスである場合にのみレンダリングされます。
「タグ」を使ってアーティクルを分類することができます。
アーティクルにタグを付けるには、tags
プロパティにタグ名を [...]
記法の配列で指定します:
---
[main]
tags = [ "travel", "europe" ]
---
<article>
...
</article>
アーティクルに一つのタグを付けるには、tags
プロパティの値を文字列で指定することができます:
---
[main]
tags = "anime"
---
<article>
...
</article>
filter
属性を使って、ページに埋め込まれたアーティクルをフィルタリングすることができます:
---
[main]
layout = "home"
---
<main>
<h1>Articles (tag:travel)</h1>
<tg:articles pattern="blog/*" filter="tag:travel"></tg:articles>
</main>
filter
属性の値は、コロン一つで区切られた文字列です。現在の仕様では、コロンの左側は常に "tag"
で、コロンの右側はタグ名です。
filter
属性を使ってアーティクルへのリンクのリストをフィルタリングすることもできます:
---
[main]
layout = "home"
---
<main>
<h1>Articles (tag:travel)</h1>
<ul>
<tg:links pattern="blog/*" filter="tag:travel">
<li>
<a href="#">
<tg:prop name="title"></tg:prop>
</a>
</li>
</tg:links>
</ul>
</main>
ページにタグを割り当てることはできません。
<tg:links>
要素に component
属性があると、その値に対応する名前の部品の内容が <tg:links>
要素の内容になります。
例えば、次のような内容の nav_link
部品があるとする。
src/components/nav_link.html
<li>
<a href="#">
<tg:prop name="title"></tg:prop>
<tg:if-complete>
<span class="text-sm">
(<tg:slot name="date"></tg:slot>)
</span>
</tg:if-complete>
</a>
</li>
この場合、<tg:links>
要素は次のように構成できます:
<tg:links component="nav_link" pattern="proposals/*"></tg:links>
上記のコードは、以下のコードとまったく同じと解釈されます。
<tg:links pattern="proposals/*">
<li>
<a href="#">
<tg:prop name="title"></tg:prop>
<tg:if-complete>
<span class="text-sm">
(<tg:slot name="date"></tg:slot>)
</span>
</tg:if-complete>
</a>
</li>
</tg:links>
ここでは、modal や carousel などの動的要素をウェブサイトに導入する方法を説明します。
HTML要素に tg:modal
属性を指定すると、その要素の子孫要素で以下の属性が利用できるようになります:
tg:open
tg:close
tg:modal
属性を持つHTML要素を「modal」と呼びます。modal は open
と close
の2つの状態を持ちます。初期状態は close
です。
modal の内部には <dialog>
要素が 1 つなければなりません。初期状態では、この要素は表示されません。ユーザーが tg:open
属性を持つ要素をクリックすると、<dialog>
要素が表示されます。逆に、ユーザーが tg:close
属性を持つ要素をクリックすると、<dialog>
要素は消えます。
以下はモーダルの例です:
<div tg:modal>
<div><button type="button" tg:open>Open</button></div>
<dialog class="rounded-2xl bg-white backdrop:bg-gray-800/80 p-4 h-[360px]">
<h2>Greetings</h2>
<p>Hello, world!</p>
<div><button type="button" tg:close>Close</button></div>
</dialog>
</div>
<dialog>
要素の直下には、「バックドロップ」と呼ばれる表示領域と同じ大きさの透明な要素が挿入されます。バックドロップは <dialog>
要素以外のすべてのページ内容を非アクティブにします。
<dialog>
要素の class
属性の値にあるクラス・トークン backdrop:bg-gray-800/80
は、背景色に半透明の濃い灰色を適用します。詳細はTailwind CSS Documentationの Dialog Backdrops を参照してください。
HTML要素に tg:toggler
属性を指定すると、その要素の子孫要素で以下の属性が利用できるようになります:
tg:when
tg:toggle
tg:toggler
属性を持つHTML要素を「toggler」と呼びます。toggler は on
と off
の2つの状態を持ちます。初期状態は off
です。
toggler 内の tg:when
属性を持つ要素は、その値が toggler の状態と一致した場合のみ表示されます。
toggler の中にある tg:toggle
属性を持つ要素をクリックまたはタップすると、toggler の状態がその属性の値に設定されます。
以下は toggler の例です:
<div tg:toggler>
<button type="button" tg:toggle="on" tg:when="off">Open</button>
<div tg:when="on">
<button type="button" tg:toggle="off">Close</button>
<p>Hello, world!</p>
</div>
</div>
この例をブラウザで見ると、最初は "Open" ボタンだけがユーザーに見えます。このボタンをクリックまたはタップすると、toggler の状態が on
に設定され、 "Open" ボタンは消え、代わりに "Close" ボタンと "Hello, world" パラグラフが表示されます。さらに "Close" ボタンをクリックまたはタップすると、初期状態に戻ります。
tg:toggle
属性の値が省略された場合、ユーザーが要素をクリックまたはタップすると toggler の状態が反転します。つまり、上記の例は次のように書き換えることができます:
<div tg:toggler>
<button type="button" tg:toggle tg:when="off">Open</button>
<div tg:when="on">
<button type="button" tg:toggle>Close</button>
<p>Hello, world!</p>
</div>
</div>
HTML要素に tg:switcher
属性を指定すると、その要素の子孫要素で以下の属性が利用できるようになります:
tg:item
tg:choose
tg:first
tg:prev
tg:next
tg:last
tg:paginator
tg:switcher
属性を持つHTML要素を「switcher」と呼びます。
switcher の内部には、必ず tg:body
属性を持つ要素、つまり「switcherボディ」がなければなりません。さらに、tg:item
属性を持つ要素、つまり「switcherアイテム」がなければなりません。
switcherアイテムには、0から順番にユニークな「index 番号」が割り当てられます。switcherは、「現在のindex番号」と呼ばれる整数値で表される状態を持っています。switcherアイテムは、そのインデックス番号が switcher の現在のインデックス番号と一致する場合のみ表示されます。
ユーザーが switcher の中の tg:choose
属性を持つ要素をクリックまたはタップすると、switcher の現在のインデックス番号がその属性の値に設定されます。
以下は switcher の例です:
<div tg:switcher>
<div tg:body>
<div tg:item>A</div>
<div tg:item>B</div>
<div tg:item>C</div>
<div tg:item>D</div>
<div tg:item>E</div>
</div>
<nav>
<button type="button" tg:choose="0">a</button>
<button type="button" tg:choose="1">b</button>
<button type="button" tg:choose="2">c</button>
<button type="button" tg:choose="3">d</button>
<button type="button" tg:choose="4">e</button>
</nav>
</div>
ボタンの tg:choose
属性の値が switcher の現在のインデックス番号と一致する場合、ユーザーがこのボタンをクリックまたはタップしても何も起こりません。
これを視覚的に現すために、tg:current-class
と tg:normal-class
属性を使用してボタンに適用されるスタイルを変更します。
<button
type="button"
tg:choose="0"
class="btn"
tg:current-class="btn-primary cursor-default"
tg:normal-class="btn-secondary">a</button>
tg:choose
属性の代わりに、tg:first
、tg:prev
、tg:next
、tg:last
などの特別な属性を使用して、スイッチャーの状態を変更するボタンを作成できます。
<div tg:switcher>
<div tg:body>
<div tg:item>A</div>
<div tg:item>B</div>
<div tg:item>C</div>
<div tg:item>D</div>
<div tg:item>E</div>
</div>
<nav>
<button type="button" tg:first>First</button>
<button type="button" tg:prev>Prev</button>
<button type="button" tg:next>Next</button>
<button type="button" tg:last>Last</button>
</nav>
</div>
switcher の現在のインデックス番号がその下限に一致する場合、ユーザーが tg:first
または tg:prev
属性のボタンをクリックまたはタップしても何も起こりません。
同様に、switcher の現在のインデックス番号がその上限と一致する場合、ユーザーが tg:next
または tg:last
属性のボタンをクリックまたはタップしても何も起こりません。
これを視覚的に現すために、tg:enabled-class
と tg:disabled-class
属性を使用してボタンに適用されるスタイルを変更します。
<button
type="button"
tg:first
class="btn"
tg:enabled-class="btn-primary"
tg:disabled-class="btn-disabled">First</button>
switcher が tg:interval
属性を持つ場合、switcher のインデックス番号は指定された間隔で1ずつ増加します(単位:ミリ秒)。
<div tg:switcher tg:interval="5000">
...
</div>
switcherアイテムにフェードイン/フェードアウトの効果を加えたい場合は、switcher の tg:transition-duration
属性にフェードイン/フェードアウトが完了するまでの時間をミリ秒単位で指定します。
<div tg:switcher tg:interval="5000" tg:transition-duration="750">
...
</div>
ユーザーが tg:choose
属性付きのボタンなどをクリックまたはタップした場合、switcher の状態は自動的に変化しなくなります。
switcher の中には、tg:paginator
属性を持つ要素があるかもしれません。この要素は、ユーザーが表示される switcherアイテムを選択するためのボタン・グループのテンプレートになります。これらのボタンを「ページネーションボタン」と呼びます。
例えば、switcher 項目の数が5つの場合、次のコード例は5つの <button>
要素を内部に持つ <nav>
要素を生成します。
<div tg:switcher>
...
<nav>
<button type="button" tg:paginator></button>
</nav>
</div>
このままでは、単なる <button></button>
要素が作られるだけなのでユーザにはには何も見えません。CSSでスタイルを指定する必要があります。例えば、次のようにすると丸いボタンが現れます。
<nav>
<button
tg:paginator
class="rounded-full w-6 h-6 mx-1 bg-gray-500"
></button>
</nav>
さらに、次のようにするとUIが洗練されます。
<nav>
<button
type="button"
tg:paginator
class="rounded-full w-6 h-6 mx-1 opacity-50"
tg:normal-class="bg-teal-400 hover:opacity-75"
tg:current-class="bg-orange-400 cursor-default"
></button>
</nav>
tg:rotator
属性を持つHTML要素を「rotator」と呼びます。
rotator は、3つの条件がある switcher とまったく同じように振る舞います:
- 現在のインデックス番号が上限と一致するときにユーザーが
tg:next
属性のボタンをクリックすると、インデックス番号はその下限に設定されます。 - 現在のインデックス番号が下限値に一致するときに
tg:prev
属性のボタンをクリックすると、インデックス番号は上限値に設定されます。 tg:interval
属性が設定されている場合、現在のインデックス番号が上限値に達すると、次回はインデックス番号が下限値に設定されます。
rotator の例です:
<div tg:rotator tg:interval="5000" tg:transition-duration="750">
<div tg:body>
<div tg:item>A</div>
<div tg:item>B</div>
<div tg:item>C</div>
<div tg:item>D</div>
<div tg:item>E</div>
</div>
<nav>
<button type="button" tg:prev>Prev</button>
<button type="button" tg:next>Next</button>
</nav>
</div>
tg:carousel
属性を持つHTML要素を「carousel」と呼びます。
これにより、ウェブサイト制作者は複数のコンテンツをスライドショーのような形式で連続して表示することができます。
carousel の内部には、必ず tg:frame
属性を持つ要素、つまり「carousel frame」がなければなりません。carouselフレームの中には、tg:body
属性を持つ要素、「carouselボディ」が必要です。また、carouselボディ の中に tg:item
属性を持つ要素、「carouselアイテム」がなければなりません。
carousel body の幅は、最初の carouselアイテム の幅の整数倍になるように自動的に計算されるので、ウェブサイト作成者が指定する必要はありません。その幅は、すべての carouselアイテム を水平に整列させるのに十分な大きさになるが、carouselフレームの overflow: hidden
スタイルのため、ウェブサイトの訪問者にはその一部しか見えません。carousel効果は、埋め込まれたJavaScriptプログラムで carousel body を左右に移動させることで実現されます。
carouselフレームの幅は、ウェブサイト作成者が調整する必要があります。通常は、carouselフレームと最初の carouselアイテム の幅を同じにします。そうすることで、carousel が静止している間、carouselフレームには1つの carouselアイテム だけが表示されます。常に複数の carousel項目 表示したい場合は、carouselフレームの幅を最初の carouselアイテム の幅より大きくしてください。
---
[style]
carousel-frame = "w-[240px] h-[180px]"
carousel-body = """
[&>div] { w-[240px] h-[180px] object-cover }
[&>div>img] { w-full h-full object-cover object-center}
"""
---
<div tg:carousel>
<div tg:frame tg:class="carousel-frame">
<div tg:body tg:class="carousel-body">
<div tg:item><img src="/images/slides/a.png"></div>
<div tg:item><img src="/images/slides/b.png"></div>
<div tg:item><img src="/images/slides/c.png"></div>
<div tg:item><img src="/images/slides/d.png"></div>
<div tg:item><img src="/images/slides/e.png"></div>
</div>
</div>
</div>
rotator と同様に、tg:prev
または tg:next
属性を持つ要素を carousel の内側に置くことで、ユーザーは carousel の状態をコントロールすることができます。
<nav>
<button type="button" tg:prev>Prev</button>
<button type="button" tg:next>Next</button>
</nav>
carousel のアイテムを一定時間ごとに自動的にローテーションさせるには、カルーセルの tg:interval
属性に正の整数を指定します。この属性で指定された値は、ミリ秒単位の時間として解釈されます。
<div tg:carousel tg:interval="3000">
carousel のローテーションにアニメーション効果を加えたい場合は、carousel の tg:transition-duration
属性を指定します。
<div tg:carousel tg:interval="3000" tg:transition-duration="500">
デフォルトでは、カルーセル・アイテムの水平方向の動きは線形、つまり等速で動きます。
carousel の動き方を微調整したい場合は、"ease-"で始まる名前のクラスをcarouselボディに指定します。
<div tg:carousel tg:transition-duration="1000">
<div tg:frame tg:class="carousel-frame">
<div tg:body class="ease-in-out" tg:class="carousel-body">
...
</div>
</div>
</div>
ease-in-out
クラスは、変化の始まりと終わりで動きを緩やかにします。詳細は Transition Timing Function を参照。
carousel に tg:transition-duration
属性が設定されている場合、carouselボディが水平方向に移動している間にユーザーが "prev "または "next "ボタンをクリック/タップしても、何の効果もありません。これを視覚的に示すには、ボタンに tg:enabled-class
と tg:disabled-class
属性を指定します。
tg:enabled-class
属性で指定されたクラス・トークンは、carouselボディが停止している時にボタンの class
属性に追加され、tg:disabled-class
属性で指定されたクラス・トークンは、carouselボディが移動している時にボタンの class
属性に追加されます。
<button
type="button"
tg:prev
class="rounded-full w-12 h-12 opacity-50"
tg:enabled-class="bg-teal-400 hover:opacity-75"
tg:disabled-class="bg-gray-400 cursor-default"
>
<span class="material-symbols-outlined">arrow_back</span>
</button>
carousel の中には、tg:paginator
属性を持つ要素があります。この要素は、ユーザーが carouselフレームの中央に表示される carouselアイテムを選択するためのボタングループのテンプレートになります。これらのボタンを「ページネーションボタン」と呼びます。
例えば、carouselアイテムの数が5つの場合、次のコード例は5つの <button>
要素を内部に持つ <nav>
要素を生成します。
<div tg:carousel>
...
<nav>
<button type="button" tg:paginator></button>
</nav>
</div>
各ページネーションボタンは carouselアイテムの1つに対応しています。
必要であれば、tg:choose
属性で carouselアイテムの番号を指定することで、個々のページネーション・ボタンをコード化することができます。
次の例は、前の例と同じように、5つの <button>
要素を持つ <nav>
要素を生成します。
<nav>
<button type="button" tg:choose="0"></button>
<button type="button" tg:choose="1"></button>
<button type="button" tg:choose="2"></button>
<button type="button" tg:choose="3"></button>
<button type="button" tg:choose="4"></button>
</nav>
各 carouselアイテムには0から始まる番号が振られていることに注意してください。
carouselフレームの中央に表示される carouselアイテムに対応するページネーションボタンに目立つスタイルを与えたい場合は、tg:normal-class
属性と tg:current-class
属性を使用します。
<nav>
<button
type="button"
tg:paginator
class="rounded-full w-6 h-6 mx-1 opacity-50"
tg:normal-class="bg-teal-400 hover:opacity-75"
tg:current-class="bg-orange-400 cursor-default"
>
</button>
</nav>
tg:normal-class
属性で指定されたクラス・トークンは、現在 carouselフレームの中央に表示されていない carouselアイテムに対応するボタンに適用され、tg:current-class
属性で指定されたクラス・トークンは、現在 carouselフレームの中央に表示されている carouselアイテムに対応するボタンに適用されます。
カルーセルに tg:transition-duration
属性が設定されている場合、carouselボディが水平方向に移動している間にユーザーがページ分割ボタンをクリック/タップしても、何の効果もありません。これを視覚的に示すには、ボタンに tg:disabled-class
属性を指定します。
<nav>
<button
type="button"
tg:paginator
class="rounded-full w-6 h-6 mx-1 opacity-50"
tg:normal-class="bg-teal-400 hover:opacity-75"
tg:current-class="bg-orange-400 cursor-default"
tg:disabled-class="bg-gray-400 cursor-default"
>
</button>
</nav>
tg:disabled-class
属性で指定されたクラス・トークンは、carouselボディが移動する際に、すべてのページネーションボタンの class
属性に追加されます。
「Scheduler」は、HTML要素とその子孫要素の class
属性を時間と共に変更する仕組みです。
以下は scheduler 設定の簡単な例です:
<div
tg:scheduler
class="w-24 mx-auto p-4 text-center text-white"
tg:init="bg-black"
tg:1000="bg-red-500"
tg:2000="bg-blue-500"
tg:3000="bg-green-500"
>
Hello, world!
</div>
tg:scheduler
属性は、この要素が scheduler よって管理されていることを宣言しています。この要素の class
属性の値は「ベースクラス」と呼ばれます。
ベースクラスに tg:init
属性の値を加えたものが、この要素の class
属性となります。この例では、<div>
要素の class
属性は初期値 "w-24 mx-auto p-4 text-center text-white bg-black"
に設定されます。
tg:
という属性と数字の並びを組み合わせることで、ウェブページが読み込まれてから一定の時間が経過した時点で、この要素の class
属性を変更します。数字の並びは経過時間をミリ秒単位で表します。
この例では、<div>
要素の class
属性の値は次のように時間と共に変化します:
- 1秒後:
w-24 mx-auto p-4 text-center text-white bg-red-500
- 2秒後:
w-24 mx-auto p-4 text-center text-white bg-blue-500
- 3秒後:
w-24 mx-auto p-4 text-center text-white bg-green-500
以下の例では、フェードイン効果を得るために scheduler を使用しています:
<div
tg:scheduler
tg:init="opacity-0"
tg:0="opacity-100 transition duration-500"
>
Fade In
</div>
初期状態では、この <div>
要素の class
属性の値は opacity-0
を含んでいるので、その内容はユーザーには見えません。ページが読み込まれた瞬間に、この <div>
要素の class
属性から opacity-0
が削除され、代わりに opacity-100
が追加されます。tg:0
属性に含まれる transition duration-500
のおかげで、opacity-100
の効果は0.5秒かけて徐々に適用されます。
「tram」は、要素と表示領域の位置関係の変化に応じて、HTML要素やその子孫要素の class
属性を変更する仕組みです。
ユーザーが tarm を配置したウェブページを上から下へスクロールすると、tarm は下から上へ進行します。tram の上端が表示領域の下端に接したとき、tram の進行度は0となります。tram の下端が表示領域の上端に接したとき、tram の進行度は100となります。
以下は tram の簡単な設定例です。:
<div tg:tram>
<div
class="w-48 h-48 mx-auto"
tg:init="bg-black"
tg:forward-50="bg-red-500"
>
</div>
</div>
この tram は内側に1つの <div>
要素を持っています。内側の要素は tg:forward-50
属性を持ち、この属性があることでこの要素が tram の「ターゲット」となります。
ターゲットの class
属性の値は「ベースクラス」と呼ばれます。最初は、ベースクラスに tg:init
属性で指定された bg-black
を加えたものが、class属性の実際の値になります。そして、tg:forward-50
という属性を指定することで、tram の進行度が50になった瞬間に、ベースクラスに bg-red-500
を加えたものがターゲットのclass属性として設定されます。50のような数字で表される tram の進行度は「トリガーポイント」と呼ばれます。
名前が tg:forward-
で始まる属性は「フォワードトリガー」と呼ばれ、名前が tg:backward-
で始まる属性は「バックトリガー」と呼ばれます。フォワードトリガーで指定されたクラストークンは、tram が前進中にそのトリガーのトリガーポイントに達すると、ターゲットの class
属性に追加されます。バックトリガーで指定されたクラストークンは、tram が後方へ移動中にそのトリガーのトリガーポイントに到達すると、ターゲットの class
属性に追加されます。
1つのターゲットに複数のトリガーを設定することが可能です。 次の例では、tram が進むにつれて背景色が黒から赤へ、そして赤から緑へと変化します。
<div tg:tram>
<div
class="w-48 h-48 mx-auto"
tg:init="bg-black"
tg:forward-25="bg-red-500"
tg:forward-50="bg-green-500"
>
</div>
</div>
色の変化を緩やかにしたい場合は、transition
とduration-1000
をベースクラスに追加します:
<div tg:tram>
<div
class="w-48 h-48 mx-auto transition duration-1000"
tg:init="bg-black"
tg:forward-50="bg-red-500"
>
</div>
</div>
こうすることで、tram が表示領域の中央まで進むと、1000ミリ秒かけて背景色が黒から赤に切り替わります。
ユーザーがウェブページを下から上へスクロールすると、tram は画面の上から下へ後退します。
tram が後方に移動している間にターゲットの class
属性を変更したい場合は、後方トリガーを指定します:
<div tg:tram>
<div
class="w-48 h-48 mx-auto"
tg:init="bg-black"
tg:forward-50="bg-red-500"
tg:backward-50="bg-black"
>
</div>
</div>
上の例では、tram が後方に移動しながら表示領域の中央に到達した瞬間に、ターゲットの背景色が赤から黒に変わります。
これまでは0から100までの「裸の」整数を使ってトリガーポイントを表現してきたが、整数に「単位」を加えることで、様々なトリガーポイントを表現することができます。
100%
は、tram の長さ(高さ)に等しい前進に相当するトリガーポイントを表します。例えば、tg:forward-50%
属性に設定されたクラストークンは、tram が表示領域の底から tram の半分の長さだけ進んだときに、ターゲットの class
属性に追加されます。
次の例では、tram のターゲットは最初は表示領域の左端より外側にあり、tram が進んでその下端が表示領域の下端に触れると、1000msかけて元の位置に移動します。
<div tg:tram class="overflow-hidden">
<div
class="w-48 h-48 mx-auto bg-black transition duration-1000"
tg:init="translate-x-[-100vw]"
tg:forward-100%="translate-x-0"
>
</div>
</div>
100vh
は、表示領域の高さに等しい進捗に相当するトリガーポイントを表します。例えば、tg:forward-50vh
属性に設定されたクラストークンは、tram の頭部が表示領域の中点と同じ高さにあるとき、ターゲットの class
属性に追加されます。
100px
は100ピクセルの進行に対応するトリガーポイントを表します。例えば、tg:forward-64px
属性は、tramが表示領域の下端から64ピクセル進んだときに適用されるべきクラストークンを値として持っています。
これらの単位には、+
または -
という接尾辞を追加することができます。接尾辞 +
は、tramの進行が表示領域の上端を基準として測定されることを意味します。例えば、50%+
はtramが表示領域の上から半分の長さだけ進んだことを示します。
接尾辞 -
は、tram の進行基準が表示領域の上端であり、トリガーポイントがビューポートの上端から後方に離れていることを意味します。例えば、64px-
は tram の先頭が表示領域の上端から64ピクセル後方にあることを示します。
tgwebは Alpine.js を使って動的なコンテンツ操作を実現しています。ただし、x-data
、 @click
、 :class
のようなAlpine.jsに由来する属性は、ウェブサイトの作者自身が使用することはできません。これらの属性が HTML ソースファイルで見つかった場合、それらは削除されます。
Teamgenikはノーコード開発プラットフォームという側面を持っています。Teamgenikで作成されたアプリケーションは「ミニアプリ」と呼ばれます。TeamgenikのSTUDIOスペースではミニアプリを作成でき、MARKETスペースではミニアプリを取得または購入できます。
ミニアプリには2つの用途があります:
- BASEスペースで独立したウィジェットとして使用できます。
- 個人またはチームのウェブサイトに埋め込むことができます。
ミニアプリはTeamgenikで公開されているウェブサイトにのみ埋め込むことができます。そのため、npx tgweb-server
コマンドで起動したウェブサーバーが配信するウェブページの上でミニアプリケーションを実行することはできません。ただし、ミニアプリのプレースホルダーを埋め込むことは可能です。その後、npx tgweb-push
コマンドでウェブサイトデータをTeamgenikにアップロードし、ミニアプリを埋め込んだウェブサイトを公開できます。
Note npx tgweb-push
コマンドはまだ使用できません。
ミニアプリのプレースホルダをウェブページに埋め込むには、ミニアプリのデフォルトロケールを指定し、site.toml
にミニアプリの名前、ID、表示名を登録する必要があります。以下にセットアップの例を示します。
default-locale = "en"
[[apps]]
name = "score_board"
id = "121d0e34-2398-4f7d-a8be-bdc549cd4332"
display-name = "Score Board"
[[apps]]
name = "players_list"
id = "70955db7-75f6-43b2-b46a-50413e43b94f"
display-name = "Players List"
ミニアプリのテキストが国際化されると、default-locale
プロパティで設定されたロケールで翻訳されます。
各ミニアプリのコンフィギュレーションの最初の行に [[apps]]
を置かなければなりません。「apps」という言葉は二重の角括弧で囲まれていることに注意してください。
name
プロパティは、設定するミニアプリを識別するための名前です。その値はどんな文字列でもよいですが、後述の <tg:app>
タグの name
属性に対応するものでなければなりません。
id
プロパティは、Teamgenik上のミニアプリに割り当てられた識別子 (ID) です。その値は UUID (Universally Unique IDentifier) の形をしています。各ミニアプリのIDは、Teamgenik BASEスペース内の「ミニアプリ」タブで確認できます。このプロパティはオプションです。
このIDは、Teamgenikで公開されているウェブサイト上で実際にミニアプリを実行するために必要ですが、単にローカル環境のウェブページにプレースホルダを埋め込み、その外観を確認したい場合は省略できます。
display-name
プロパティの値はプレースホルダー内に表示される文字列です。このプロパティが省略された場合、代わりに name
プロパティの値が使用されます。
ミニアプリのプレースホルダーをウェブページに埋め込むには、<tg:app>
要素を使います。その name
属性の値は、site.toml
で説明したミニアプリの name
プロパティと一致しなければなりません。以下は <tg:app>
要素の使用例です。
<tg:app name="score_board"></tg:app>
<tg:app>
要素は expanded
属性を持つことができます。この属性はミニアプリの表示モードを制御します。
ミニアプリには、標準モードと拡張モードの2つの表示モードがあります。標準モードのミニアプリのレイアウトは横幅300ピクセルに最適化され、拡張モードのミニアプリのレイアウトは横幅640ピクセルに最適化されます。
expanded
属性はブーリアン属性で、この属性が存在すればミニ・アプリは拡張モードで表示され、そうでなければ標準モードで表示されます。以下は expanded
属性の使用例です:
<tg:app name="score_board" expanded></tg:app>
ミニアプリのプレースホルダー自体には特定の幅と高さがないことに注意してください。通常、プレースホルダーの幅は次のように <div>
要素で囲んで固定します:
<div class="w-[300px]">
<tg:app name="score_board"></tg:app>
</div>
以下のように最小の高さを指定することで、ミニアプリケーションのコンテンツが変更されたときに、ウェブページのレイアウトがぐらつくのを防ぐことができます。
<div class="w-[300px] min-h-[450px]">
<tg:app name="score_board"></tg:app>
</div>
ブラウザの表示幅が 640 ピクセル以上かどうかに応じて表示モードを切り替えるには、次のように 2 つのプレースホルダーを配置し、適切な Tailwind CSS クラス トークンのセットを含む <div>
要素でそれぞれを囲みます。
<div class="w-[300px] sm:hidden">
<tg:app name="score_board"></tg:app>
</div>
<div class="w-[640px] hidden sm:block">
<tg:app name="score_board" expanded></tg:app>
</div>
sm:hidden
クラスは、ブラウザーの表示幅が 640 ピクセルを超える場合に要素を非表示にします。hidden sm:block
クラスは、ブラウザの表示幅が 640 ピクセル未満の場合に要素を非表示にします。
ここで説明した Tailwind CSS クラス トークンの詳細については、次のページを参照してください:
- https://tailwindcss.com/docs/width#arbitrary-values
- https://tailwindcss.com/docs/min-height#arbitrary-values
- https://tailwindcss.com/docs/responsive-design
- https://tailwindcss.com/docs/display
<tg:plugin>
要素を用いて HubSpot のフォームをあなたのウェブサイトに埋め込むことができます。
HubSpot にアカウント登録して HubSpot フォームを作成すると、次のようなコード例が提供されます。
<script charset="utf-8" type="text/javascript" src="//js.hsforms.net/forms/embed/v2.js"></script>
<script>
hbspt.forms.create({
portalId: "01234567",
formId: "01234567-89ab-cdef-0123-456789abcdef"
});
</script>
portalId:
の右にある 8 桁の数列がポータル ID で、formID:
の右にある長さ 36 の文字列がフォーム ID です。このとき、次のように記載すると HubSpot フォームを埋め込むことができます。
<tg:plugin name="hubspot" portal-id="01234567" form-id="01234567-89ab-cdef-0123-456789abcdef"></tg:plugin>
src
ディレクトリに site.toml
という名前のファイルを作成すると、サイトレベルでプロパティの値を設定することができます。 ここで設定された値は、各ページのフロントマターで設定されたプロパティのデフォルト値となります。
src/site.toml
title = "No title"
layout = "common"
[data]
current-year = 2023
画像や音声ファイルのURLを <meta>
要素の content
属性に埋め込むには %{...}
表記を使います:
[data]
icon-url = "%{images/icons/default.png}"
theme-url = "%{audios/our_theme.mp3}"
具体的な使用例については elementsを参照してください。
ページのフロントマターで特定のプロパティに値が設定されていない場合、tgwebは以下の順序で値を検索します。
- 可能な場合は、そのラッパーのフロントマター
- 可能な場合は、そのレイアウトのフロントマター
- 可能な場合は、
sites.toml
例えば、あるページのフロントマターで、次のようにカスタムプロパティ x
の値が "a"
と設定されているとします。
[data]
x = "a"
また、ラッパーのフロントマターが以下のような設定になっているとします:
[data]
x = "b"
y = "c"
この場合、このページのカスタムプロパティ x
の値は "a"
であり、カスタムプロパティ y
の値は "c"
です。
さらに、このページに適用されているレイアウトのフロントマターが以下のような設定になっているとします:
[data]
z = "d"
この場合、このページのプロパティ z
の値は "d"
となります。
また更に、sites.toml
が以下のような設定になっているとします:
[data]
s = "e"
この場合、このページのカスタムプロパティ s
の値は "e"
となります。
カスタムプロパティだけでなく、title
のような定義済みのプロパティも継承されます。また、"meta"、"http-equiv"、"meta-property"、"link" セクションに属するプロパティも継承されます。ただし、"style" セクションに属するプロパティは 継承されません 。
アーティクルが独立した HTML ドキュメントとしてレンダリングされるとき、プロパティの継承のメカニズムはページと全く同じです。
アーティクルがページ、セグメント、レイアウトに埋め込まれるとき、もしあれば、アーティクルを囲むラッパーと site.toml
からプロパティを継承しますが、そのアーティクルを埋め込むページ、セグメント、レイアウトからは継承しません。
ラッパーやレイアウトがページやアーティクルに適用されるとき、そのラッパーやレイアウトの <tg:prop>
や <tg:data>
要素に置き換えられる値は、ページやアーティクルが持つプロパティの値です。
例えば、あるページに次のようなフロントマターがあるとします。
[data]
x = "a"
そして、そのラッパーが次のようなフロントマターとHTML断片を持っているとします。:
---
[data]
x = "b"
y = "c"
---
<header>
<tg:data name="x"></tg:data>
<tg:data name="y"></tg:data>
</header>
<tg:content></tg:content>
すると、そこから生成されるHTML文書の <header>
要素のテキスト内容は、"b c "ではなく、"a c "となります。
セグメントがページに埋め込まれるとき、そのセグメントの <tg:prop>
要素と <tg:data>
要素に置き換えられる値は、ページが持つプロパティの値です。
同様に、セグメントがレイアウトの中にあるとき、そのセグメント内の <tg:prop>
と <tg:data>
要素に置き換えられる値は、メインテンプレート(ページまたはアーティクル)のプロパティの値です。
部品がページ、アーティクル、セグメント、ラッパー、レイアウトに埋め込まれるとき、そのテンプレートの <tg:prop>
と <tg:data>
要素に置き換えられる値は、メインテンプレート(ページまたはアーティクル)のプロパティの値です。
<title>
要素の内容は、ページやアーティクルのテンプレートから以下のルールで決定されます:
- もしあれば、
title
プロパティの値 - もしあれば、最初の
<h1>
要素のテキスト内容 - もしあれば、最初の
<h2>
要素のテキスト内容 - もしあれば、最初の
<h3>
要素のテキスト内容 - もしあれば、最初の
<h4>
要素のテキスト内容 - もしあれば、最初の
<h5>
要素のテキスト内容 - もしあれば、最初の
<h6>
要素のテキスト内容 "No Title"
ページやアーティクルのフロントマターで title
プロパティの値が設定されていなくても、ラッパーやレイアウト、あるいは sites.toml
から継承している場合があることに注意してください。
以下のテンプレートから生成されるHTMLドキュメントのタイトルは "Greeting" になります:
---
[main]
title = "Greeting"
---
<body>
<h1>Welcome!</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</body>
文字列 "Welcome!" は次のテンプレートからタイトルとして抽出されます。
<body>
<h1>Welcome!</h1>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</body>
次のテンプレートがHTMLドキュメントとしてレンダリングされる場合、そのタイトルは "No Title" となります:
<body>
<div class="text-2xl">Welcome!</div>
<div class="bg-green-300 p-4">
<p>Hello, world!</p>
</div>
</body>
<head>
要素内の <meta>
要素は "meta"、"http-equiv"、"meta-property" セクションに属するプロパティの値によって生成されます。
生成されたHTML文書の <head>
要素には、必ず <meta charset="utf-8">
要素が含まれていることに注意してください。
"meta.name" セクションに属するプロパティに値を設定することで、name
属性を持つ <meta>
要素を生成できます:
[meta.name]
viewport = "width=device-width, initial-scale=1"
theme-color = "#2da0a8"
description = "Description"
robots = "index,follow"
generator = "Teamgenik"
上記のようにプロパティの値を設定すると、以下の <meta>
要素が生成される。
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#2da0a8">
<meta name="description" content="Description">
<meta name="robots" content="index,follow">
<meta name="generator" content="Teamgenik">
同じ名前の <meta>
要素を複数生成したい場合は、次のように記述します。
[meta.name]
googlebot = [ "index,follow", "notranslate" ]
上記は以下の <meta>
要素を生成します。
<meta name="googlebot" content="index,follow">
<meta name="googlebot" content="notranslate">
"http-equiv" セクションに属するプロパティに値を設定することで、http-equiv
属性を持つ <meta>
要素を生成できます。
[meta.http-equiv]
content-security-policy = "default-src 'self'"
x-dns-prefetch-control = "off"
上記の設定により、以下の <meta>
要素が生成されます:
<meta http-equiv="content-security-policy" content="default-src 'self'">
<meta http-equiv="x-dns-prefetch-control" content="off">
Teamgenikはこれらのパスを適切にURLに変換します。
[meta.property]
"fb:app_id" = "1234567890abcde"
"fb:article_style" = "default"
"fb:use_automatic_ad_placement" = "true"
"op:markup_version" = "v1.0"
"al:ios:app_name" = "App Links"
"al:android:app_name" = "App Links"
上記の設定は、以下の <meta>
要素を生成します。
<meta property="fb:app_id" content="1234567890abcde">
<meta property="fb:article_style" content="default">
<meta property="fb:use_automatic_ad_placement" content="true">
<meta property="op:markup_version" content="v1.0">
<meta property="al:ios:app_name" content="App Links">
<meta property="al:android:app_name" content="App Links">
プロパティの値は、${...}
記法を使って <meta>
要素の content
属性に埋め込むことができます:
[meta.property]
"og:url" = "${url}"
"og:title" = "${title}"
"og:description" = "${meta.name.description}"
"meta" セクションに属するプロパティの値を参照するには、プロパティ名の前に meta.
を加えます。
画像や音声ファイルのURLは、%{...}
記法を使って <meta>
要素の content
属性に埋め込むことができます:
[meta.property]
"og:image" = "%{/images/icon.png}"
"og:audio" = "%{/audios/theme.mp3}"
<head>
要素内の <link>
要素は、"link" セクション、"links" セクション、"stylesheets" セクションに属するプロパティの値によって生成されます。
[link]
canonical = "https://example.com/"
license = "%{/copyright.html}"
[[links]]
blocking = "render"
href = "example.woff2"
as = "font"
[[links]]
rel = "preload"
href = "my_font.woff2"
as = "font"
type = "font/woff2"
crossorigin = "anonymous"
[[stylesheets]]
main = true
secondary = false
"foo/bar" = true
上記は以下の <link>
要素を生成します。
<link rel="canonical" href="https://example.com/">
<link rel="license" href="http://localhost:3000/copyright.html">
<link blocking="render" href="example.woff2" as="font">
<link rel="preload" href="myFont.woff2" as="font" type="font/woff2" crossorigin="anonymous">
<link rel="stylesheet" type="text/css" href="/main.css">
<link rel="stylesheet" type="text/css" href="/foo/bar.css">
ウェブサイトがTeamgenikで公開されると、%{...}
表記から生成されたURLは適切に変換されます。
次の <link>
要素は常にhead要素の中に挿入されます。
<link href="/css/tailwind.css" rel="stylesheet">
npx tgweb-archive
コマンドを用いて作られた .zip
ファイルを TeamGenik にインポートした場合、site.toml
およびフロントマターに含まれる "stylesheets" セクションは無視されます。
すなわち、TeamGenik 上で管理・公開する Web サイトにおいては、独自の CSS ファイルを使用することはできません。
<head>
要素内の <script>
要素は、"scripts" セクションに属するプロパティの値によって生成されます。
[[scripts]]
main = true
secondary = false
"foo/bar" = true
上記は以下の <link>
要素を生成します。
<script type="text/javascript" src="/main.js"></script>
<script type="text/javascript" src="/foo/bar.js"></script>
次の <script>
要素は常にhead要素内に挿入されます。
<script src="/js/tgweb_utilities.js" defer></script>
<script src="/js/alpine.min.js" defer></script>
<script type="module" src="/js/tgweb_lottie_player.js"></script>
<script src="/reload/reload.js" defer></script>
npx tgweb-archive
コマンドを用いて作られた .zip
ファイルを TeamGenik にインポートした場合、site.toml
およびフロントマターに含まれる "scripts" セクションは無視されます。
すなわち、TeamGenik 上で管理・公開する Web サイトにおいては、独自の JavaScript ファイルを使用することはできません。
あなたが Cloudflare Pages を利用中であれば、npx tgweb-deploy
コマンドを用いて dist
ディレクトリに作られた配布用アセットを特定のアプリケーション(別名:プロジェクト)にアップロードできます。
npx tgweb-deploy
コマンドを利用する場合、複数サイト構造を採用する必要があります。
すなわち、sites
ディレクトリの下に各サイトのためのディレクトリを作り、その下にある src
ディレクトリにソースコードを配置する必要があります。
また、各サイトのためのディレクトリ名と Cloudflare Pages アプリケーションの名前は一致している必要があります。
例えば、Cloudflare Pages アプリケーションの名前が my_site
であれば、sites/my_site/src
ディレクトリに ソースコードを置いてください。
my_site
アプリケーションにデプロイメントする手順は、次の通りです:
npx tgweb-dist sites/my_site
コマンドを実行。npx tgweb-deploy sites/my_site
コマンドを実行。
ただし、npx tgweb-deploy
コマンドを利用するためには、準備作業として npx wrangler login
コマンドで Cloudflare にログインする必要があります。このコマンドを実行すると、自動的にブラウザが開きます。
ここで「Allow」ボタンをクリックすると npx tgweb-deploy
コマンドが利用可能になります。
npx tgweb-preview
コマンドを利用すると Cloudflare Pages のプレビュー環境に配布用アセットがデプロイされます。
<tg:animation>
: アニメーション<tg:article>
: ページやセグメントにアーティクルを埋め込む<tg:articles>
: ページに複数のアーティクルを埋め込む<tg:app>
: ミニアプリの埋め込み<tg:component>
: 部品を埋め込む<tg:content>
: レイアウトの追加, ラッパーの追加<tg:data>
: カスタムプロパティ<tg:if-complete>
:<tg:if-complete>
<tg:if-current>
:<tg:link>、<tg:if-current>、<tg:label>
<tg:if-embedded>
:<tg:if-embedded> と <tg:unless-embedded>
<tg:insert>
: スロットとインサート<tg:label>
:<tg:link>、<tg:if-current>、<tg:label>
<tg:link>
:<tg:link>、<tg:if-current>、<tg:label>
<tg:links>
: リンクリスト<tg:plugin>
: プラグイン<tg:prop>
: Embedding predefined property values in a template<tg:segment>
: Embedding segments into a page<tg:shared-component>
: 共有部品<tg:slot>
: スロットとインサート<tg:symbol>
: シンボル<tg:unless-embedded>
:<tg:if-embedded> と <tg:unless-embedded>
tg:1000
, etc.: Schedulertg:backward-*
: Tramtg:body
: Switcher, Rotator, Carouseltg:carousel
: Carouseltg:choose
: Switcher, Rotatortg:class
: スタイルエイリアスの定義tg:close
: Modaltg:current-class
: Switcher, Rotator, Carouseltg:disabled-class
: Carouseltg:enabled-class
: Carouseltg:first
: Switcher, Rotatortg:frame
: Carouseltg:forward-*
: Tramtg:init
: Scheduler, Tramtg:interval
: Rotator, Carouseltg:item
: Carouseltg:last
: Switcher, Rotatortg:modal
: Modaltg:next
: Switcher, Rotatortg:normal-class
: Switcher, Rotator, Carouseltg:open
: Modaltg:paginator
: Switcher/Paginator, Carousel/Paginatortg:prev
: Switcher, Rotatortg:rotator
: Rotatortg:switcher
: Switchertg:scheduler
: Schedulertg:toggle
: Togglertg:toggler
: Togglertg:tram
: Tramtg:transition-duration
: Carouseltg:when
: Toggler
See TODO.md.
tgweb is MIT licensed.