ブランド追跡ページの構築ガイド
お客様が配送業者の追跡リンクをクリックするたびに、自社サイトから離れてしまいます。これはエンゲージメント、ブランディング、サポート負荷の軽減にとって大きな機会損失です。ブランド追跡ページを用意すれば、お客様を自社ドメインに留めながら、より良い体験を提供できます。実際に構築してみましょう。
なぜブランド追跡ページが必要なのか
配送業者の追跡ページの問題点
お客様を配送業者の追跡ページに誘導すると、以下のようなことが起こります。
- 自社のブランド体験から離脱してしまう
- 御社のプロモーションではなく、配送業者の広告が表示される
- 御社のサポートチームに簡単に連絡できない
- 追跡ページが分かりにくい場合、配送業者ではなく御社が不満の対象になる
ブランド追跡ページのメリット
| メリット | 効果 |
|---|---|
| お客様を自社サイトに留める | ページ再訪率 +15% |
| サポートチケットの削減 | 追跡関連の問い合わせ -40% |
| アップセルの機会 | 関連商品の提案が可能 |
| ブランドの一貫性 | プロフェッショナルな体験の提供 |
| データ収集 | 顧客エンゲージメントの把握 |
アーキテクチャ
お客様が追跡リンクをクリック
↓
自社のブランド追跡ページ
↓
JavaScriptが自社APIから追跡データを取得
↓
自社APIがWhereParcelを呼び出し → 標準化されたデータを返却
↓
追跡タイムライン+ブランド要素をレンダリング
ステップ1:バックエンドAPIエンドポイント
WhereParcelへのリクエストをプロキシするAPIエンドポイントを作成します。
// routes/tracking.js
const express = require('express');
const router = express.Router();
router.get('/api/track/:trackingNumber', async (req, res) => {
const { trackingNumber } = req.params;
try {
// 追跡番号で注文を検索
const order = await db.orders.findByTracking(trackingNumber);
// WhereParcelから追跡データを取得
const tracking = await fetch('https://api.whereparcel.com/v2/track', {
method: 'POST',
headers: {
'Authorization': `Bearer ${process.env.WHEREPARCEL_API_KEY}:${process.env.WHEREPARCEL_SECRET_KEY}`,
'Content-Type': 'application/json',
},
body: JSON.stringify({
trackingItems: [{ carrier: order?.carrier || 'auto', trackingNumber }],
}),
}).then(r => r.json());
res.json({
order: order ? {
id: order.id,
items: order.items.map(i => ({
name: i.name,
image: i.imageUrl,
quantity: i.quantity,
})),
} : null,
tracking: tracking.data,
});
} catch (error) {
res.status(500).json({ error: 'Unable to fetch tracking data' });
}
});
module.exports = router;
ステップ2:追跡ページのHTML構造
追跡ページの基本構造を作成します。
<!DOCTYPE html>
<html lang="ja">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>配送状況の確認 | YourStore</title>
<style>
/* ステップ4でスタイルを定義 */
</style>
</head>
<body>
<header class="tracking-header">
<a href="/" class="logo">
<img src="/logo.svg" alt="YourStore" />
</a>
<a href="/support" class="support-link">お困りですか?</a>
</header>
<main class="tracking-container">
<!-- 検索フォーム(URLに追跡番号がない場合) -->
<div id="search-section" class="search-section">
<h1>配送状況を確認する</h1>
<form id="tracking-form">
<input
type="text"
id="tracking-input"
placeholder="追跡番号を入力してください"
required
/>
<button type="submit">追跡する</button>
</form>
</div>
<!-- 結果表示 -->
<div id="results-section" class="results-section" style="display:none">
<div id="status-banner" class="status-banner"></div>
<div id="order-items" class="order-items"></div>
<div id="tracking-timeline" class="tracking-timeline"></div>
</div>
<!-- ローディング表示 -->
<div id="loading" class="loading" style="display:none">
<div class="spinner"></div>
<p>追跡情報を取得しています...</p>
</div>
</main>
<footer class="tracking-footer">
<p>© 2026 YourStore. All rights reserved.</p>
</footer>
<script src="/tracking.js"></script>
</body>
</html>
ステップ3:フロントエンドJavaScript
追跡データの取得とレンダリングを実装します。
// public/tracking.js
const STATUS_CONFIG = {
pending: { label: '準備中', icon: 'clock', color: '#6b7280' },
info_received: { label: '情報受付済み', icon: 'file', color: '#3b82f6' },
picked_up: { label: '集荷済み', icon: 'package', color: '#3b82f6' },
in_transit: { label: '配送中', icon: 'truck', color: '#f59e0b' },
out_for_delivery: { label: 'お届け中', icon: 'map-pin', color: '#8b5cf6' },
delivered: { label: 'お届け済み', icon: 'check-circle', color: '#10b981' },
exception: { label: '例外発生', icon: 'alert', color: '#ef4444' },
};
// URLから追跡番号を取得
const urlParams = new URLSearchParams(window.location.search);
const trackingNumber = urlParams.get('tn');
if (trackingNumber) {
document.getElementById('tracking-input').value = trackingNumber;
fetchTracking(trackingNumber);
}
// フォーム送信
document.getElementById('tracking-form').addEventListener('submit', (e) => {
e.preventDefault();
const tn = document.getElementById('tracking-input').value.trim();
if (tn) {
// リロードせずにURLを更新
window.history.pushState({}, '', `?tn=${tn}`);
fetchTracking(tn);
}
});
async function fetchTracking(trackingNumber) {
const loading = document.getElementById('loading');
const results = document.getElementById('results-section');
const search = document.getElementById('search-section');
loading.style.display = 'flex';
results.style.display = 'none';
try {
const response = await fetch(`/api/track/${trackingNumber}`);
const data = await response.json();
if (!response.ok) throw new Error(data.error);
renderResults(data);
results.style.display = 'block';
} catch (error) {
results.innerHTML = `
<div class="error-message">
<h2>追跡情報が見つかりませんでした</h2>
<p>追跡番号をご確認のうえ、もう一度お試しください。</p>
</div>
`;
results.style.display = 'block';
} finally {
loading.style.display = 'none';
}
}
function renderResults(data) {
const { tracking, order } = data;
const config = STATUS_CONFIG[tracking.status] || STATUS_CONFIG.pending;
// ステータスバナー
document.getElementById('status-banner').innerHTML = `
<div class="status-icon" style="background: ${config.color}20; color: ${config.color}">
${config.label}
</div>
<h2>${getStatusMessage(tracking.status)}</h2>
${tracking.estimatedDelivery
? `<p class="eta">お届け予定日: ${formatDate(tracking.estimatedDelivery)}</p>`
: ''}
`;
// 注文商品(データがある場合)
if (order?.items?.length) {
document.getElementById('order-items').innerHTML = `
<h3>ご注文の商品</h3>
<div class="items-grid">
${order.items.map(item => `
<div class="item-card">
<img src="${item.image}" alt="${item.name}" />
<span>${item.name}</span>
<span class="qty">x${item.quantity}</span>
</div>
`).join('')}
</div>
`;
}
// タイムライン
document.getElementById('tracking-timeline').innerHTML = `
<h3>配送履歴</h3>
<div class="timeline">
${tracking.events.map((event, i) => `
<div class="timeline-event ${i === 0 ? 'latest' : ''}">
<div class="timeline-dot"></div>
<div class="timeline-content">
<div class="event-time">${formatDateTime(event.timestamp)}</div>
<div class="event-status">${event.description}</div>
${event.location ? `<div class="event-location">${event.location}</div>` : ''}
</div>
</div>
`).join('')}
</div>
`;
}
function getStatusMessage(status) {
const messages = {
pending: 'ご注文の準備を進めています',
picked_up: '荷物が集荷されました',
in_transit: '荷物を配送中です',
out_for_delivery: '本日お届け予定です',
delivered: '荷物がお届け済みです',
exception: '配送に問題が発生しています',
};
return messages[status] || '荷物を追跡しています';
}
function formatDate(dateStr) {
return new Date(dateStr).toLocaleDateString('ja-JP', {
year: 'numeric', month: 'long', day: 'numeric', weekday: 'long',
});
}
function formatDateTime(dateStr) {
return new Date(dateStr).toLocaleString('ja-JP', {
month: 'short', day: 'numeric', hour: 'numeric', minute: '2-digit',
});
}
ステップ4:CSSスタイリング
自社ブランドに合わせたスタイリングを適用します。
:root {
--brand-primary: #4f46e5;
--brand-secondary: #818cf8;
--text-primary: #1f2937;
--text-secondary: #6b7280;
--bg-primary: #ffffff;
--bg-secondary: #f9fafb;
--border: #e5e7eb;
}
.tracking-container {
max-width: 640px;
margin: 0 auto;
padding: 2rem 1rem;
}
.status-banner {
text-align: center;
padding: 2rem;
background: var(--bg-secondary);
border-radius: 12px;
margin-bottom: 2rem;
}
.status-icon {
display: inline-block;
padding: 0.5rem 1rem;
border-radius: 9999px;
font-weight: 600;
font-size: 0.875rem;
margin-bottom: 0.75rem;
}
.timeline {
position: relative;
padding-left: 2rem;
}
.timeline::before {
content: '';
position: absolute;
left: 7px;
top: 0;
bottom: 0;
width: 2px;
background: var(--border);
}
.timeline-event {
position: relative;
padding-bottom: 1.5rem;
}
.timeline-dot {
position: absolute;
left: -2rem;
top: 4px;
width: 16px;
height: 16px;
border-radius: 50%;
background: var(--border);
border: 3px solid var(--bg-primary);
}
.timeline-event.latest .timeline-dot {
background: var(--brand-primary);
}
.event-time {
font-size: 0.8125rem;
color: var(--text-secondary);
}
.event-status {
font-weight: 500;
color: var(--text-primary);
}
.event-location {
font-size: 0.875rem;
color: var(--text-secondary);
}
ステップ5:エンゲージメント要素の追加
追跡ページの価値を最大化するための仕組みを追加しましょう。
関連商品のアップセル
function renderRecommendations(order) {
if (!order) return;
return `
<section class="recommendations">
<h3>こちらの商品もおすすめです</h3>
<div class="product-grid">
${getRelatedProducts(order.items).map(product => `
<a href="${product.url}" class="product-card">
<img src="${product.image}" alt="${product.name}" />
<span class="product-name">${product.name}</span>
<span class="product-price">${product.price}</span>
</a>
`).join('')}
</div>
</section>
`;
}
フィードバックの収集
お届け完了後に、配送体験についてのフィードバックを収集します。
if (tracking.status === 'delivered') {
resultsHTML += `
<section class="feedback-section">
<h3>配送体験はいかがでしたか?</h3>
<div class="rating-buttons">
<button onclick="submitRating(5)">とても良い</button>
<button onclick="submitRating(4)">良い</button>
<button onclick="submitRating(3)">普通</button>
<button onclick="submitRating(2)">改善の余地あり</button>
</div>
</section>
`;
}
SEOのメリット
ブランド追跡ページはSEOにも好影響をもたらします。
- サイト滞在時間の増加 — お客様が追跡を何度も確認するため
- 直帰率の低下 — お客様が自社ドメインに留まるため
- 内部リンクの強化 — 商品ページ、ブログ、サポートページへの導線を設置可能
- ブランド権威の向上 — プロフェッショナルな追跡体験が信頼構築につながる
まとめ
ブランド追跡ページは、最もROIの高いプロジェクトの一つです。サポートチケットの削減、顧客エンゲージメントの向上、アップセル機会の創出、そしてプロフェッショナルな体験の提供を実現できます。WhereParcelのAPIが追跡データを提供するので、御社はフロントエンド体験の構築に専念できます。
関連リソース:
- WhereParcel APIドキュメント — 追跡データの取得
- Webhookの設定 — リアルタイム更新の受信
- EC配送通知 — 通知システムの全体像