在 AEM 中編輯外部 SPA editing-external-spa-within-aem

決定要在外部SPA和AEM之間進行何種程度的整合時,請考量您通常必須能夠編輯和檢視AEM中的SPA。

概觀 overview

本檔案說明將獨立SPA上傳到AEM執行個體、新增內容的可編輯區段及啟用編寫的建議步驟。

先決條件 prerequisites

先決條件很簡單。

  • 請確定AEM的執行個體正在本機執行。

  • 使用AEM專案原型建立基礎AEM SPA專案。

    • Forms是AEM專案的基礎,已更新以包括外部SPA。
    • 對於此檔案中的範例,Adobe是使用WKND SPA專案的起點。
  • 隨時準備要整合的可使用外部React SPA。

上傳SPA至AEM專案 upload-spa-to-aem-project

首先,您必須將外部SPA上傳至AEM專案。

  1. /ui.frontend專案資料夾中的src取代為React應用程式的src資料夾。
  2. /ui.frontend/package.json檔案中的應用程式package.json中包含任何其他相依性。
  3. /public資料夾中包含任何自訂專案。
  4. 包含任何新增至/public/index.html檔案的內嵌指令碼或樣式。

設定遠端SPA configure-remote-spa

現在,外部SPA屬於AEM專案的一部分,必須在AEM中進行設定。

包含AdobeSPA SDK套件 include-spa-sdk-packages

若要利用AEM SPA功能,需依賴下列三個套件。

@adobe/aem-spa-page-model-manager套件提供用於初始化模型管理員和從AEM執行個體擷取模型的API。 然後,此模型可用於使用來自@adobe/aem-react-editable-components@adobe/aem-spa-component-mapping的API來轉譯AEM元件。

安裝 installation

執行下列npm命令,以便安裝必要的套件。

npm install --save @adobe/aem-spa-component-mapping @adobe/aem-spa-page-model-manager @adobe/aem-react-editable-components

ModelManager初始化 model-manager-initialization

在應用程式轉譯之前,必須初始化ModelManager以處理AEM ModelStore的建立。

此初始化必須在應用程式的src/index.js檔案中或在呈現應用程式根目錄的任何位置完成。

若要進行此初始化,您可以使用ModelManager提供的initializationAsync API。

下列熒幕擷圖顯示如何在簡單的React應用程式中啟用ModelManager的初始化。 唯一的限制是必須在ReactDOM.render()之前呼叫initializationAsync

初始化ModelManager

在此範例中,已初始化ModelManager並建立空的ModelStore

initializationAsync可選擇接受options物件做為引數:

  • path — 初始化時,會擷取定義路徑上的模型並儲存在ModelStore中。 如有需要,可使用此路徑在初始化時擷取rootModel
  • modelClient — 允許提供負責擷取模型的自訂使用者端。
  • model — 作為引數傳遞的model物件通常在使用SSR🔗的時填入。

AEM Authorable Leaf元件 authorable-leaf-components

  1. 建立/識別已為其建立可授權的React元件的AEM元件。 在此範例中,它使用WKND專案的文字元件。

    WKND文字元件

  2. 在SPA中建立簡單的React文字元件。 在此範例中,已建立包含下列內容的新檔案Text.js

    Text.js

  3. 建立組態物件,讓您指定啟用AEM編輯所需的屬性。

    建立設定物件

    • 在AEM編輯器中開啟時,resourceType必須將該React元件對應到AEM元件並啟用編輯。
  4. 使用包裝函式withMappable

    使用withMappable

    此包裝函式將React元件對應到設定中指定的AEM resourceType,並在AEM編輯器中開啟時啟用編輯功能。 對於獨立元件,它也會擷取特定節點的模型內容。

    note note
    NOTE
    在此範例中,有不同版本的元件: AEM包裝和未包裝的React元件。 明確使用元件時,必須使用包裝的版本。 當元件是頁面的一部分時,您可以繼續使用預設元件,就像目前在SPA編輯器中完成的一樣。
  5. 呈現元件中的內容。

    文字元件的JCR屬性在AEM中會顯示如下。

    文字元件屬性

    這些值會作為屬性傳遞至已建立的AEMText React元件,並可用於呈現內容。

    code language-javascript
    import React from 'react';
    import { withMappable } from '@adobe/aem-react-editable-components';
    
    export const TextEditConfig = {
        // Empty component placeholder label
        emptyLabel:'Text',
        isEmpty:function(props) {
           return !props || !props.text || props.text.trim().length < 1;
        },
        // resourcetype of the AEM counterpart component
        resourceType:'wknd-spa-react/components/text'
    };
    
    const Text = ({ text }) => (<div>{text}</div>);
    
    export default Text;
    
    export const AEMText = withMappable(Text, TextEditConfig);
    

    以下是AEM設定完成時元件的顯示方式。

    code language-javascript
    const Text = ({ cqPath, richText, text }) => {
       const richTextContent = () => (
          <div className="aem_text" id={cqPath.substr(cqPath.lastIndexOf('/') + 1)} data-rte-editelement dangerouslySetInnerHTML={{__html: text}}/>
       );
       return richText ? richTextContent() : (<div className="aem_text">{text}</div>);
    };
    
    note note
    NOTE
    在此範例中,已對轉譯的元件進行進一步自訂,以符合現有的文字元件。 這與AEM中的製作無關。

將可編寫的元件新增至頁面 add-authorable-component-to-page

建立可授權的React元件後,即可在整個應用程式中使用這些元件。

以一個頁面為例,您必須從WKND SPA專案新增文字。 在此範例中,您想要顯示文字「Hello World!」 於/content/wknd-spa-react/us/en/home.html

  1. 決定要顯示的節點路徑。

    • pagePath:此範例中包含節點的頁面/content/wknd-spa-react/us/en/home
    • itemPath:頁面中節點的路徑,在此範例中為root/responsivegrid/text
      • 由頁面上包含專案的名稱組成。

    節點的路徑

  2. 在頁面的所需位置新增元件。

    新增元件至頁面

    AEMText元件可新增至頁面中設定為屬性之pagePathitemPath值的必要位置。 pagePath是強制屬性。

驗證在AEM上編輯文字內容 verify-text-edit

現在請在執行中的AEM例項上測試元件。

  1. aem-guides-wknd-spa目錄執行以下Maven命令,以便您可以建置專案並將其部署到AEM。
mvn clean install -PautoInstallSinglePackage
  1. 在您的AEM執行個體上,瀏覽至http://<host>:<port>/editor.html/content/wknd-spa-react/us/en/home.html

在AEM中編輯SPA

AEMText元件現在可在AEM上編寫。

AEM可編寫頁面 aem-authorable-pages

  1. 識別要新增以在SPA中編寫的頁面。 此範例使用/content/wknd-spa-react/us/en/home.html

  2. 為可編寫的頁面元件建立檔案(例如Page.js)。 使用@adobe/cq-react-editable-components中提供的頁面元件。

  3. 在區段AEM可編寫的分葉元件中重複步驟4。 在元件上使用包裝函式withMappable

  4. 如同先前所做的,將MapTo套用至頁面內所有子元件的AEM資源型別。

    code language-javascript
    import { Page, MapTo, withMappable } from '@adobe/aem-react-editable-components';
    import Text, { TextEditConfig } from './Text';
    
    export default withMappable(Page);
    
    MapTo('wknd-spa-react/components/text')(Text, TextEditConfig);
    
    note note
    NOTE
    在此範例中,會使用未包裝的React文字元件,而非先前建立的已包裝的AEMText。 原因是當元件屬於頁面/容器且不是獨立元件時,容器會負責遞回對應元件。 此外,每個子項也不需要啟用編寫功能和其他包裝函式。
  5. 若要在SPA中新增可編寫的頁面,請依照新增可編寫的元件至頁面一節中的相同步驟操作。 在這裡,您可以略過itemPath屬性。

驗證AEM上的頁面內容 verify-page-content

若要確認頁面可以編輯,請依照區段確認在AEM上編輯文字內容中的相同步驟操作。

在AEM中編輯頁面

該頁面現在可在AEM上編輯,並包含版面容器和子文字元件。

虛擬分葉元件 virtual-leaf-components

在先前的範例中,您已將元件新增至具有現有AEM內容的SPA。 但在某些情況下,內容尚未在AEM中建立,但必須由內容作者稍後新增。 為因應這種情況,前端開發人員可以在SPA內的適當位置新增元件。 在AEM的編輯器中開啟這些元件時,會顯示預留位置。 內容作者將內容新增到這些預留位置後,節點會在JCR結構中建立並持續保留內容。 建立的元件允許與獨立葉元件相同的操作集。

在此範例中,您正在重複使用先前建立的AEMText元件。 您想要在WKND首頁上的現有文字元件下方新增文字。 新增的元件與一般葉元件相同。 不過,itemPath可以更新為必須新增新元件的路徑。

因為新元件必須新增到root/responsivegrid/text的現有文字下方,所以新路徑為root/responsivegrid/{itemName}

<AEMText
 pagePath='/content/wknd-spa-react/us/en/home'
 itemPath='root/responsivegrid/text_20' />

新增虛擬元件後,TestPage元件看起來像以下專案。

正在測試虛擬元件

NOTE
請確定AEMText元件已在組態中設定其resourceType,以便您能夠啟用此功能。

您現在可以依照驗證AEM上文字內容的編輯一節中的步驟,將變更部署到AEM。 目前不存在的text_20節點會顯示預留位置。

aem 中的text_20節點

內容作者更新此元件時,會在/content/wknd-spa-react/us/en/homeroot/responsivegrid/text_20建立新的text_20節點。

text20節點

要求和限制 limitations

新增虛擬分葉元件有幾項需求和部分限制。

  • pagePath屬性是建立虛擬元件的必要屬性。
  • pagePath中的路徑所提供的頁面節點必須存在於AEM專案中。
  • 必須在itemPath中提供要建立的節點名稱。
  • 可在任何層級建立元件。
    • 如果您在上一個範例中提供itemPath='text_20',則會直接在頁面下建立新節點,即/content/wknd-spa-react/us/en/home/jcr:content/text_20
  • 透過itemPath提供時,建立新節點的節點路徑必須有效。
    • 在此範例中,root/responsivegrid必須存在,才能在那裡建立新節點text_20
  • 僅支援建立分葉元件。 未來版本將支援虛擬容器和頁面。

虛擬容器 virtual-containers

即使尚未在AEM中建立對應的容器,也支援新增容器的功能。 概念與方法類似虛擬分葉元件

前端開發人員可以在AEM內的適當位置新增容器元件,這些元件在SPA的編輯器中開啟時會顯示預留位置。 然後,作者可以將元件及其內容新增到容器中,容器會在JCR結構中建立所需的節點。

例如,如果容器存在於/root/responsivegrid,而開發人員想要新增子容器:

容器位置

newContainer尚未存在於AEM中。

在AEM中編輯包含此元件的頁面時,會顯示容器的空白預留位置,以供作者新增內容。

容器預留位置

JCR 中的容器位置

一旦作者將子元件新增至容器後,新容器節點就會以JCR結構中的對應名稱建立。

包含內容的容器

含有JCR 內容的容器

現在可以根據作者的需要將更多元件和內容新增到容器中,且變更會持續存在。

要求和限制 container-limitations

新增虛擬容器有幾項需求和幾項限制。

  • 決定可新增哪些元件的原則繼承自父容器。

  • 要建立的容器的直接父系必須存在於AEM中。

    • 如果AEM容器中存在容器root/responsivegrid,則可提供路徑root/responsivegrid/newContainer來建立新容器。
    • 但是root/responsivegrid/newContainer/secondNewContainer是不可能的。
  • 一次只能建立一個新層級的元件。

其他自訂 additional-customizations

如果您依照前面的範例進行,現在可以在AEM內編輯外部SPA。 不過,您外部SPA還有其他方面可以進一步自訂。

根節點ID root-node-id

依預設,您可以假設React應用程式呈現在元素識別碼spa-rootdiv內。 如有需要,可自訂此語法。

例如,假設您有一個SPA,應用程式在元素識別碼rootdiv內呈現。 此語法必須反映在三個檔案中。

  1. 在React應用程式的index.js中(或呼叫ReactDOM.render()的位置)

    index.js檔案 中的ReactDOM.render()

  2. 在React應用程式的index.html

    應用程式的index.html

  3. 在AEM應用程式的頁面元件本文中,透過兩個步驟:

    1. 為頁面元件建立body.html

    建立body.html檔案

    1. 在新的body.html檔案中新增根元素。

    將根元素新增至body.html

使用路由編輯React SPA editing-react-spa-with-routing

如果外部React SPA應用程式有多個頁面,它可以使用路由來決定要轉譯的頁面/元件。 基本的使用案例是將目前作用中的URL與為路由提供的路徑進行比對。 若要在啟用路由的應用程式上啟用編輯,必須轉換要比對的路徑以容納AEM的特定資訊。

在下列範例中,您有一個包含兩個頁面的簡單React應用程式。 要呈現的頁面是根據提供給路由器的路徑與作用中URL相符來決定。 例如,如果您在mydomain.com/test,則會轉譯TestPage

在外部SPA中進行路由

若要在AEM內啟用此範例SPA的編輯功能,需要下列步驟。

  1. 識別將做為AEM根的層級。

    • 如需範例,請將wknd-spa-react/us/en視為SPA的根目錄。 此根目錄表示該路徑之前的所有內容都只有AEM頁面/內容。
  2. 在所需層級建立頁面。

    • 在此範例中,要編輯的頁面為mydomain.com/testtest在應用程式的根路徑中。 在AEM中建立頁面時,也必須保留此根路徑。 因此,您可以在先前步驟中定義的根層級建立頁面。
    • 建立的新頁面必須與要編輯的頁面同名。 在此範例中,對於mydomain.com/test,建立的新頁面必須是/path/to/aem/root/test
  3. 在SPA路由中新增協助程式。

    • 建立的頁面尚無法在AEM中轉譯預期的內容。 原因是因為路由器預期路徑為/test,而AEM作用中路徑為/wknd-spa-react/us/en/test。 為了適應AEM特定的URL部分,您必須在SPA端新增一些協助程式。

    路由協助程式

    • 可以使用@adobe/cq-spa-page-model-manager提供的toAEMPath協助程式。 它會轉換為路由提供的路徑,以便在AEM例項上開啟應用程式時包含AEM的特定部分。 它接受三個引數:

      • 路由所需的路徑
      • 編輯SPA的AEM執行個體的來源URL
      • AEM上的專案根目錄,如第一個步驟所決定
    • 這些值可設為環境變數,以獲得更多彈性。

  4. 驗證是否在AEM中編輯頁面。

    • 將專案部署到AEM並導覽至已建立的test頁面。 頁面內容現在已呈現,且AEM元件可供編輯。

框架限制 framework-limitations

RemotePage元件預期實作會提供資產資訊清單,例如GitHub🔗上的webpack-manifest-plugin。 不過,RemotePage元件僅通過測試可用於React架構(以及透過remote-page-next元件的Next.js),因此不支援從其他架構(例如Angular)從遠端載入應用程式。

其他資源 additional-resources

下列參考資料可協助您瞭解AEM內容中的SPA。

recommendation-more-help
fbcff2a9-b6fe-4574-b04a-21e75df764ab