開発(汎用) developing-generic
統合フレームワークには、API を備えた統合レイヤーが含まれています。これにより、特定の e コマースエンジンに依存しない e コマース機能用の AEM コンポーネントを作成できます。さらに、内部 CRX データベースを使用したり、e コマースシステムを組み込んだりして、製品データを AEM に取り込むこともできます。
この統合レイヤーを使用するために、すぐに使える AEM コンポーネントがいくつか用意されています。現時点では、以下のようなものがあります。
- 製品表示コンポーネント
- 買い物かご
- プロモーションと割引券
- カタログおよびセクションのブループリント
- チェックアウト
- 検索
検索については、Adobe Experience Manager(AEM)検索やサードパーティ検索、またはこれらを組み合わせた検索を行うための統合フックが提供されています。
e コマースエンジンの選択 ecommerce-engine-selection
e コマースフレームワークは任意の e コマースソリューションと組み合わせて使用できますが、使用するエンジンは AEM によって識別される必要があります(AEM の汎用エンジンを使用する場合でも同様です)。
-
e コマースエンジンは、
CommerceService
インターフェイスをサポートする OSGi サービスです。- エンジンは、
commerceProvider
サービスプロパティによって区別できます。
- エンジンは、
-
AEM では、
CommerceService
とProduct
に対してResource.adaptTo()
をサポートしています。-
adaptTo
実装は、リソースの階層内でcq:commerceProvider
プロパティを探します。- 見つかった場合は、その値を使用してコマースサービスの検索をフィルタリングします。
- 見つからなかった場合は、最上位のコマースサービスが使用されます。
-
厳密に型指定されたリソースに
cq:commerceProvider
を追加できるように、cq:Commerce
Mixin が使用されます。
-
-
適切なコマースファクトリ定義を参照するために、
cq:commerceProvider
プロパティも使用されます。- 例えば、Geometrixx という値を持つ
cq:commerceProvider
プロパティは、Geometrixx-Outdoors 向け Day CQ コマースファクトリ(com.adobe.cq.commerce.hybris.impl.GeoCommerceServiceFactory
)の OSGi 設定に関連付けられます(ここでも、commerceProvider
パラメーターの値はgeometrixx
となります)。 - ここで、その他のプロパティも設定できます(適切かつ使用可能な場合)。
- 例えば、Geometrixx という値を持つ
標準の AEM インストールでは、例えば次のような特定の実装が必要です。
cq:commerceProvider = geometrixx
例 example
/etc/commerce/products/geometrixx-outdoors
+ cq:commerceProvider = geometrixx
+ adobe-logo-shirt
+ cq:commerceType = product
+ price = 12.50
+ adobe-logo-shirt_S
+ cq:commerceType = variant
+ size = S
+ adobe-logo-shirt_XL
+ cq:commerceType = variant
+ size = XL
+ price = 14.50
/apps/geometrixx-outdoors/components/product
セッション処理 session-handling
顧客の買い物かごに関連する情報を保存するためのセッションです。
CommerceSession は、
-
買い物かご を所有しています。
-
追加や削除などを実行します。
-
買い物かごについての様々な計算を実行します。
commerceSession.getProductPriceInfo(Product product, Predicate filter)
-
-
注文 データの永続性を管理します。
CommerceSession.getUserContext()
-
updateOrder(Map<String, Object> delta)
を使用して、配送の詳細情報を取得または更新できます。 -
支払 処理の接続も管理します。
-
フルフィルメント の接続も管理します。
アーキテクチャ architecture
製品とバリアントのアーキテクチャ architecture-of-product-and-variants
1 つの製品に複数のバリエーションが存在する場合があります。例えば、カラーやサイズで異なるバリエーションが存在する場合があります。製品では、バリエーションを構成するプロパティを定義する必要があります。このようなプロパティを バリアント軸 と呼びます。
ただし、すべてのプロパティがバリアント軸になるわけではありません。バリエーションは、他のプロパティにも影響を与えることがあります。例えば、価格はサイズに依存することがあります。このようなプロパティは、買い物客が選択できないので、バリアント軸とは見なされません。
各製品やバリアントはリソースによって表現されるので、リポジトリノードに 1 対 1 でマップされます。当然のことながら、特定の製品やバリアントはそのパスによって一意に識別できます。
すべての製品リソースは、Product API
で表現できます。製品 API の呼び出しのほとんどはバリエーションごとに固有ですが(ただし、バリエーションは 1 つの上位のものから共有される値を継承する場合があります)、一連のバリエーションを一覧で表示する呼び出しもあります(getVariantAxes()
や getVariants()
など)。
Product.getVariantAxes()
の戻り値で決定されます。- 汎用実装の場合、AEM が製品データのプロパティ(
cq:productVariantAxes
)からバリアント軸を読み取ります。
-
size
-
さらにもう 1 つ
この追加バリアントは、製品リファレンスの
variationAxis
プロパティ(Geometrixx Outdoors の場合、通常はcolor
)を使用して選択されます。
製品リファレンスと PIM データ product-references-and-pim-data
一般的には以下のようになります。
-
PIM データは
/etc
の下に配置されます。 -
製品リファレンスは
/content
の下に配置されます。
製品バリエーションと製品データノードの間には 1 対 1 のマッピングが必要です。
製品リファレンスには、各バリエーションを表すノードも必要ですが、すべてのバリエーションを表す必要はありません。例えば、製品に S、M、L のバリエーションがある場合、製品データは次のようになります。
etc
commerce
products
shirt
shirt-s
shirt-m
shirt-l
「Big and Tall」カタログには次のバリエーションだけが含まれます。
content
big-and-tall
shirt
shirt-l
最後に、製品データを使用するための要件はありません。すべての製品データをカタログ内のリファレンスの配下に配置できますが、実際には、すべての製品データを複製せずに複数のカタログを用意することはできません。
API
com.adobe.cq.commerce.api.Product インターフェイス com-adobe-cq-commerce-api-product-interface
public interface Product extends Adaptable {
public String getPath(); // path to specific variation
public String getPagePath(); // path to presentation page for all variations
public String getSKU(); // unique ID of specific variation
public String getTitle(); // shortcut to getProperty(TITLE)
public String getDescription(); // shortcut to getProperty(DESCRIPTION)
public String getImageUrl(); // shortcut to getProperty(IMAGE_URL)
public String getThumbnailUrl(); // shortcut to getProperty(THUMBNAIL_URL)
public <T> T getProperty(String name, Class<T> type);
public Iterator<String> getVariantAxes();
public boolean axisIsVariant(String axis);
public Iterator<Product> getVariants(VariantFilter filter) throws CommerceException;
}
com.adobe.cq.commerce.api.VariantFilter com-adobe-cq-commerce-api-variantfilter
/**
* Interface for filtering variants and AxisFilter provided as common implementation
*
* The <code>VariantFilter</code> is used to filter variants,
* for example, when using {@link Product#getVariants(VariantFilter filter)}.
*/
public interface VariantFilter {
public boolean includes(Product product);
}
/**
* A {@link VariantFilter} for filtering variants by the given
* axis and value. The following example returns a list of
* variant products that have a value of <i>blue</i> on the
* <i>color</i> axis.
*
* <p>
* <code>product.getVariants(new AxisFilter("color", "blue"));</code>
*/
public class AxisFilter implements VariantFilter {
private String axis;
private String value;
public AxisFilter(String axis, String value) {
this.axis = axis;
this.value = value;
}
/**
* {@inheritDoc}
*/
public boolean includes(Product product) {
ValueMap values = product.adaptTo(ValueMap.class);
if(values != null) {
String v = values.get(axis, String.class);
return v != null && v == value;
}
return false;
}
}
-
一般的なストレージメカニズム
-
製品ノードは nt:unstructured です。
-
製品ノードは、次のいずれかになります。
-
リファレンス。製品データは他の場所に保存されています。
- 製品リファレンスには、
productData
プロパティが含まれます。このプロパティは、製品データ(一般的には/etc/commerce/products
の下にあります)を指します。 - 製品データは階層化されています。製品属性は、製品データノードの上位ノードから継承されます。
- 製品リファレンスには、ローカルプロパティも含めることができます。このようなプロパティは、製品データ内で指定されるプロパティを上書きします。
- 製品リファレンスには、
-
製品自体。
productData
プロパティはありません。- すべてのプロパティをローカルに保持している(そして productData プロパティを含まない)製品ノードは、製品属性を自身の上位ノードから直接継承します。
-
-
-
AEM 汎用製品構造
- それぞれのバリアントには、独自のリーフノードが必要です。
- 製品インターフェイスは、製品とバリアントの両方を表しますが、関連リポジトリノードはそれぞれに固有です。
- 製品ノードは、製品属性とバリアント軸を記述します。
例 example-1
+ banyan_shirt
- cq:commerceType = product
- cq:productAttributes = [jcr:title, jcr:description, size, price, color]
- cq:productVariantAxes = [color, size]
- jcr:title = Banyan Shirt
- jcr:description = Flowery, all-cotton shirt.
- price = 14.00
+ banyan_shirt_s
- cq:commerceType = variant
- size = S
+ banyan_shirt_s_red
- cq:commerceType = variant
- color = red
+ banyan_shirt_s_blue
- cq:commerceType = variant
- color = blue
+ banyan_shirt_m
- cq:commerceType = variant
- size = M
+ banyan_shirt_m_red
- cq:commerceType = variant
- color = red
+ banyan_shirt_m_blue
- cq:commerceType = variant
- color = blue
+ banyan_shirt_l
- cq:commerceType = variant
- size = L
+ banyan_shirt_l_red
- cq:commerceType = variant
- color = red
+ banyan_shirt_l_blue
- cq:commerceType = variant
- color = blue
+ banyan_shirt_xl
- cq:commerceType = variant
- size = XL
- price = 18.00
買い物かごのアーキテクチャ architecture-of-the-shopping-cart
コンポーネント
-
買い物かごは、
CommerceSession:
によって管理されます。CommerceSession
は追加や削除などを実行します。CommerceSession
は、買い物かごに対する様々な計算も実行します。CommerceSession
は、トリガーされた割引券とプロモーションも買い物かごに適用します。
-
買い物かごに直接関連はしませんが、
CommerceSession
はカタログの価格情報も提供する必要があります(価格を管理しているので)。-
価格には、以下の複数の修飾子があります。
- 数量割引。
- 様々な通貨。
- 付加価値税(VAT)課税対象、および付加価値税免除
-
修飾子は、以下のインターフェイスを使用して変更することができます。
int CommerceSession.getQuantityBreakpoints(Product product)
String CommerceSession.getProductPrice(Product product)
-
ストレージ
-
ストレージ
- AEM 汎用の場合、買い物かごは ClientContext に保存されます。
パーソナライズ機能
-
パーソナライズ機能は、常に ClientContext で行われます。
-
買い物かごの ClientContext
/version/
は、すべてのケースで作成されます。- 製品は、
CommerceSession.addCartEntry()
メソッドを使用して追加する必要があります。
- 製品は、
-
次の図は、ClientContext に格納される買い物かご情報の例を示しています。
チェックアウトのアーキテクチャ architecture-of-checkout
買い物かごと注文データ
CommerceSession
は、次の 3 つの要素を管理します。
-
買い物かごのコンテンツ
買い物かごのコンテンツのスキーマは、次の API で決定されます。
code language-java public void addCartEntry(Product product, int quantity); public void modifyCartEntry(int entryNumber, int quantity); public void deleteCartEntry(int entryNumber);
-
価格
価格のスキーマも、API によって決められています。
code language-java public String getCartPreTaxPrice(); public String getCartTax(); public String getCartTotalPrice(); public String getOrderShipping(); public String getOrderTotalTax(); public String getOrderTotalPrice();
-
注文の詳細
しかし、注文の詳細は API によって決められていません 。
code language-java public void updateOrderDetails(Map<String, String> orderDetails); public Map<String, String> getOrderDetails(); public void submitOrder();
送料の計算
-
注文フォームには、多くの場合、複数の配送オプション(および価格)を提示する必要があります。
-
価格は、品目や注文の詳細(重量や配送先住所など)に基づく場合があります。
-
CommerceSession
はすべての依存関係にアクセスするので、製品価格と同じ方法で扱うことができます。CommerceSession
は送料を管理します。updateOrder(Map<String, Object> delta)
を使用すると、配送の詳細情報を取得または更新できます。
検索の定義 search-definition
標準のサービス API モデルに従って、e コマースプロジェクトは、個々のコマースエンジンで実装できる一連の検索関連の API を提供します。
e コマースプロジェクトには、次の場所にデフォルトの検索コンポーネントが含まれています。
/libs/commerce/components/search
これは、検索 API を使用して、選択したコマースエンジンに対してクエリを実行します。詳しくは、e コマースエンジンの選択を参照してください。
検索 API search-api
コアプロジェクトは、次のような汎用/ヘルパークラスを複数提供しています。
-
CommerceQuery
検索クエリの記述に使用します(クエリテキスト、現在のページ、ページサイズ、並べ替え、選択されているファセットについての情報を含みます)。検索 API を実装するすべての e コマースサービスは、検索を実行するために、このクラスのインスタンスを受け取ります。
CommerceQuery
は、リクエストオブジェクト(HttpServletRequest
)からインスタンス化できます。 -
FacetParamHelper
1 つの静的メソッド(
toParams
)を提供するユーティリティクラス。このメソッドを使用して、ファセットのリストと 1 つのトグル値からGET
パラメーター文字列を生成します。これは、UI 側で、ユーザーがハイパーリンクをクリックすると対応する値に切り替わるように、各ファセットの値ごとのハイパーリンクを表示する必要がある場合に役立ちます。この切り替えは、選択された場合はクエリから削除され、選択されていない場合は追加されるということです。これは、複数値または単一値のファセットの処理、値のオーバーライドなどのすべてのロジックを処理します。
検索 API のエントリポイントは、CommerceService#search
オブジェクトを返す CommerceResult
メソッドです。このトピックについて詳しくは、API ドキュメントを参照してください。
プロモーションと割引券の作成 developing-promotions-and-vouchers
-
割引券:
-
割引券はページベースのコンポーネントで、Web サイトコンソールを使用して作成および編集され、次の場所に保存されます。
/content/campaigns
-
割引券は、以下を提供します。
- 割引券コード(買い物客が買い物かごに入力)。
- 割引券ラベル(買い物客が買い物かごに入力した後に表示される)。
- プロモーションパス(割引券が適用されるアクションを定義)。
-
割引券には、独自の開始日時や終了日時は設定されておらず、親キャンペーンの開始日時や終了日時を使用します。
-
外部コマースエンジンでも割引券を提供することができます。以下が最低限必要です。
- 割引券コード
isValid()
メソッド
-
割引券 コンポーネント(
/libs/commerce/components/voucher
)は次のものを提供します。- 割引券管理用のレンダラー。買い物かごに現在入っている割引券があれば表示します。
- 割引券を管理(追加/削除)するための編集ダイアログボックス(フォーム)。
- 割引券を買い物かごに追加/買い物かごから削除するために必要なアクション。
-
-
プロモーション:
-
プロモーションは、ページベースのコンポーネントで、Web サイトコンソールを使用して作成/編集され、次の場所に保存されます。
/content/campaigns
-
プロモーションは、以下の項目を提供します。
- 優先度
- プロモーションハンドラーパス
-
プロモーションをキャンペーンに関連付けて、有効/無効の日付/回数を定義できます。
-
プロモーションをエクスペリエンスに関連付けて、セグメントを定義できます。
-
エクスペリエンスに関連付けられていないプロモーションは、単独ではトリガーされませんが、割引券によってトリガーすることはできます。
-
プロモーションコンポーネント(
/libs/commerce/components/promotion
)には、次のものが含まれます。- プロモーション管理用のレンダラーとダイアログ
- プロモーションハンドラー固有の設定パラメーターをレンダリングおよび編集するためのサブコンポーネント
-
次の 2 つのプロモーションハンドラーが、デフォルトで提供されています。
DiscountPromotionHandler
買い物かご全体に絶対価格による割引またはパーセンテージ割引を適用します。PerfectPartnerPromotionHandler
パートナー製品も買い物かごに入っている場合に、製品の絶対価格による割引またはパーセンテージ割引を適用します。
-
ClientContext
SegmentMgr
がセグメントを解決し、ClientContextCartMgr
がプロモーションを解決します。少なくとも 1 つの解決されたセグメントを対象とする各プロモーションがトリガーされます。- トリガーされたプロモーションは、買い物かごを再計算するために、AJAX 呼び出しによってサーバーに返送されます。
- トリガーされたプロモーション(および追加された割引券)は、ClientContext パネルにも表示されます。
-
割引券の買い物かごへの追加または買い物かごからの削除は、CommerceSession
API を使用して実行されます。
/**
* Apply a voucher to this session's cart.
*
* @param code the voucher's code
* @throws CommerceException
*/
public void addVoucher(String code) throws CommerceException;
/**
* Remove a voucher that was previously added with {@link #addVoucher(String)}.
*
* @param code the voucher's code
* @throws CommerceException
*/
public void removeVoucher(String code) throws CommerceException;
/**
* Get a list of vouchers that were added to this cart via {@link #addVoucher(String)}.
*
* @throws CommerceException
*/
public List<Voucher> getVouchers() throws CommerceException;
このように、CommerceSession
は、割引券が存在するかどうか、割引券を適用できるかどうかをチェックします。このチェックは、特定の条件を満たした場合(例えば、買い物かごの合計価格が 1 万円を超える場合など)にのみ適用できる割引券に対して実行されます。何らかの理由で割引券を適用できない場合は、addVoucher
メソッドが例外をスローします。また、CommerceSession
によって、割引券の追加または削除後に買い物かごの価格の更新もおこないます。
Voucher
は、以下のフィールドを含む、Bean のようなクラスです。
- 割引券コード
- 簡単な説明
- 割引のタイプと値を示す関連プロモーションの参照
提供される AbstractJcrCommerceSession
によって割引券を適用できます。クラス getVouchers()
が返す割引券は、(特に)次のプロパティを持つ jcr:content ノードを含んだ cq:Page
のインスタンスです。
-
sling:resourceType
(String) -commerce/components/voucher
である必要があります。 -
jcr:title
(String) - 割引券の説明用 -
code
(String) - この割引券を適用するためにユーザーが入力する必要があるコード。 -
promotion
(String) - 適用されるプロモーション(例:/content/campaigns/geometrixx-outdoors/article/10-bucks-off
)
プロモーションハンドラーは、買い物かごを変更する OSGi サービスです。買い物かごは、PromotionHandler
インターフェイスで定義される複数のフックをサポートします。
/**
* Apply promotion to a cart line item. The method returns a discount
* <code>PriceInfo</code> instance or <code>null</code> if no discount
* was applied.
* @param commerceSession The commerce session
* @param promotion The configured promotion
* @param cartEntry The cart line item
* @return A discounted <code>PriceInfo</code> or <code>null</code>
*/
public PriceInfo applyCartEntryPromotion(CommerceSession commerceSession,
Promotion promotion, CartEntry cartEntry)
throws CommerceException;
/**
* Apply promotion to an order. The method returns a discount
* <code>PriceInfo</code> instance or <code>null</code> if no discount
* was applied.
* @param commerceSession The commerce session
* @param promotion The configured promotion
* @return A discounted <code>PriceInfo</code> or <code>null</code>
*/
public PriceInfo applyOrderPromotion(CommerceSession commerceSession, Promotion promotion)
throws CommerceException;
public PriceInfo applyShippingPromotion(CommerceSession commerceSession, Promotion promotion)
throws CommerceException;
/**
* Allows a promotion handler to define a custom, author-oriented message for a promotion.
* The {@link com.adobe.cq.commerce.common.promotion.PerfectPartnerPromotionHandler}, for instance,
* uses this to list the qualifying pairs of products in the current cart.
* @param commerceSession
* @param promotion
* @return A message to display
* @throws CommerceException
*/
public String getMessage(SlingHttpServletRequest request, CommerceSession commerceSession, Promotion promotion) throws CommerceException;
/**
* Informs the promotion handler that something under the promotions root has been edited, and the handler
* should invalidate any caches it might be keeping.
*/
public void invalidateCaches();
次の 3 つのプロモーションハンドラーが、デフォルトで提供されています。
DiscountPromotionHandler
買い物かご全体に絶対価格による割引またはパーセンテージ割引を適用します。PerfectPartnerPromotionHandler
製品パートナーも買い物かごに入っている場合に、製品の絶対価格による割引またはパーセンテージ割引を適用します。FreeShippingPromotionHandler
は送料無料を適用します。