Componente personalizzato
- Argomenti:
- Core Components
- API
Creato per:
- Principiante
- Sviluppatore
Questa esercitazione descrive la creazione end-to-end di un componente AEM Byline
personalizzato che visualizza il contenuto creato in una finestra di dialogo ed esplora lo sviluppo di un modello Sling per incapsulare la logica di business che popola l'HTL del componente.
Prerequisiti
Esaminare gli strumenti e le istruzioni necessari per configurare un ambiente di sviluppo locale.
Progetto iniziale
Consulta il codice della riga di base su cui si basa l’esercitazione:
-
Controlla il ramo
tutorial/custom-component-start
da GitHub$ cd aem-guides-wknd $ git checkout tutorial/custom-component-start
-
Implementa la base di codice in un’istanza AEM locale utilizzando le tue competenze Maven:
$ mvn clean install -PautoInstallSinglePackage
NOTESe utilizzi AEM 6.5 o 6.4, aggiungi il profiloclassic
a qualsiasi comando Maven.$ mvn clean install -PautoInstallSinglePackage -Pclassic
Puoi sempre visualizzare il codice finito su GitHub o estrarre il codice localmente passando al ramo tutorial/custom-component-solution
.
Obiettivo
- Come creare un componente AEM personalizzato
- Scopri come incapsulare la logica di business con i modelli Sling
- Come utilizzare un modello Sling da uno script HTL
Cosa intendi creare
In questa parte dell’esercitazione WKND, viene creato un componente Byline che viene utilizzato per visualizzare informazioni scritte sul collaboratore di un articolo.
Componente Byline
L’implementazione del componente Byline include una finestra di dialogo che raccoglie il contenuto del nome e un modello Sling personalizzato che recupera i dettagli come:
- Nome
- Immagine
- Occupazioni
Crea componente Byline
Innanzitutto, crea la struttura del nodo del componente Byline e definisci una finestra di dialogo. Rappresenta il componente nell’AEM e definisce implicitamente il tipo di risorsa del componente in base alla sua posizione nel JCR.
La finestra di dialogo espone l’interfaccia che gli autori di contenuti possono fornire. Per questa implementazione, il componente Immagine del componente core WCM dell'AEM viene utilizzato per gestire l'authoring e il rendering dell'immagine di Byline, pertanto deve essere impostato come sling:resourceSuperType
di questo componente.
Creare una definizione di componente
-
Nel modulo ui.apps, passa a
/apps/wknd/components
e crea una cartella denominatabyline
. -
Nella cartella
byline
, aggiungere un file denominato.content.xml
-
Popolare il file
.content.xml
con quanto segue:<?xml version="1.0" encoding="UTF-8"?> <jcr:root xmlns:sling="http://sling.apache.org/jcr/sling/1.0" xmlns:cq="http://www.day.com/jcr/cq/1.0" xmlns:jcr="http://www.jcp.org/jcr/1.0" jcr:primaryType="cq:Component" jcr:title="Byline" jcr:description="Displays a contributor's byline." componentGroup="WKND Sites Project - Content" sling:resourceSuperType="core/wcm/components/image/v2/image"/>
Il file XML riportato sopra fornisce la definizione del componente, inclusi il titolo, la descrizione e il gruppo.
sling:resourceSuperType
punta acore/wcm/components/image/v2/image
, che è il Componente immagine core.
Creare lo script HTL
-
All'interno della cartella
byline
, aggiungere un filebyline.html
, responsabile della presentazione HTML del componente. È importante assegnare al file lo stesso nome della cartella, in quanto diventa lo script predefinito utilizzato da Sling per il rendering di questo tipo di risorsa. -
Aggiungere il codice seguente a
byline.html
.<!--/* byline.html */--> <div data-sly-use.placeholderTemplate="core/wcm/components/commons/v1/templates.html"> </div> <sly data-sly-call="${placeholderTemplate.placeholder @ isEmpty=true}"></sly>
byline.html
è rivisitato più tardi, una volta creato il modello Sling. Lo stato corrente del file HTL consente al componente di visualizzare lo stato vuoto nell’Editor pagina di AEM Sites quando viene trascinato e rilasciato sulla pagina.
Creare la definizione della finestra di dialogo
Quindi, definisci una finestra di dialogo per il componente Byline con i campi seguenti:
- Nome: un campo di testo che indica il nome del collaboratore.
- Immagine: riferimento alla biografia del collaboratore.
- Occupazioni: elenco di occupazioni attribuito al collaboratore. Le occupazioni devono essere ordinate alfabeticamente in ordine crescente (dalla a alla z).
-
Nella cartella
byline
, creare una cartella denominata_cq_dialog
. -
All'interno di
byline/_cq_dialog
, aggiungere un file denominato.content.xml
. Definizione XML per la finestra di dialogo. Aggiungi il seguente XML:<?xml version="1.0" encoding="UTF-8"?> <jcr:root xmlns:sling="http://sling.apache.org/jcr/sling/1.0" xmlns:cq="http://www.day.com/jcr/cq/1.0" xmlns:jcr="http://www.jcp.org/jcr/1.0" xmlns:nt="http://www.jcp.org/jcr/nt/1.0" jcr:primaryType="nt:unstructured" jcr:title="Byline" sling:resourceType="cq/gui/components/authoring/dialog"> <content jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/container"> <items jcr:primaryType="nt:unstructured"> <tabs jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/tabs" maximized="{Boolean}false"> <items jcr:primaryType="nt:unstructured"> <asset jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}false"/> <metadata jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> <properties jcr:primaryType="nt:unstructured" jcr:title="Properties" sling:resourceType="granite/ui/components/coral/foundation/container" margin="{Boolean}true"> <items jcr:primaryType="nt:unstructured"> <columns jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/fixedcolumns" margin="{Boolean}true"> <items jcr:primaryType="nt:unstructured"> <column jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/container"> <items jcr:primaryType="nt:unstructured"> <name jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/form/textfield" emptyText="Enter the contributor's name to display." fieldDescription="The contributor's name to display." fieldLabel="Name" name="./name" required="{Boolean}true"/> <occupations jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/form/multifield" fieldDescription="A list of the contributor's occupations." fieldLabel="Occupations" required="{Boolean}false"> <field jcr:primaryType="nt:unstructured" sling:resourceType="granite/ui/components/coral/foundation/form/textfield" emptyText="Enter an occupation" name="./occupations"/> </occupations> </items> </column> </items> </columns> </items> </properties> </items> </tabs> </items> </content> </jcr:root>
Queste definizioni dei nodi di dialogo utilizzano Sling Resource Merger per controllare quali schede di dialogo sono ereditate dal componente
sling:resourceSuperType
, in questo caso il componente immagine Componenti core.
Creare la finestra di dialogo dei criteri
Seguendo lo stesso approccio utilizzato per la creazione della finestra di dialogo, crea una finestra di dialogo Criterio (precedentemente nota come finestra di dialogo per progettazione) per nascondere i campi indesiderati nella configurazione Criterio ereditati dal componente Immagine dei Componenti core.
-
Nella cartella
byline
, creare una cartella denominata_cq_design_dialog
. -
In
byline/_cq_design_dialog
, creare un file denominato.content.xml
. Aggiorna il file con il seguente XML. È più semplice aprire.content.xml
e copiarvi/incollare l'XML sottostante.<?xml version="1.0" encoding="UTF-8"?> <jcr:root xmlns:sling="http://sling.apache.org/jcr/sling/1.0" xmlns:granite="http://www.adobe.com/jcr/granite/1.0" xmlns:cq="http://www.day.com/jcr/cq/1.0" xmlns:jcr="http://www.jcp.org/jcr/1.0" xmlns:nt="http://www.jcp.org/jcr/nt/1.0" jcr:primaryType="nt:unstructured" jcr:title="Byline" sling:resourceType="cq/gui/components/authoring/dialog"> <content jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <tabs jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <properties jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <content jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <decorative jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> <altValueFromDAM jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> <titleValueFromDAM jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> <displayCaptionPopup jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> <disableUuidTracking jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> </items> </content> </items> </properties> <features jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <content jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <accordion jcr:primaryType="nt:unstructured"> <items jcr:primaryType="nt:unstructured"> <orientation jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> <crop jcr:primaryType="nt:unstructured" sling:hideResource="{Boolean}true"/> </items> </accordion> </items> </content> </items> </features> </items> </tabs> </items> </content> </jcr:root>
La base per la finestra di dialogo dei criteri XML precedente è stata ottenuta dal componente core Immagine.
Come nella configurazione della finestra di dialogo, Sling Resource Merger viene utilizzato per nascondere campi irrilevanti che vengono altrimenti ereditati da
sling:resourceSuperType
, come mostrato dalle definizioni del nodo con la proprietàsling:hideResource="{Boolean}true"
.
Distribuire il codice
-
Sincronizza le modifiche in
ui.apps
con l'IDE o utilizzando le abilità Maven.
Aggiungere il componente a una pagina
Per mantenere la semplicità e l'attenzione sullo sviluppo dei componenti AEM, aggiungiamo il componente Byline nello stato corrente a una pagina dell'articolo per verificare che la definizione del nodo cq:Component
sia corretta. Inoltre, per verificare che l’AEM riconosca la definizione del nuovo componente e che la finestra di dialogo del componente funzioni per l’authoring.
Aggiungere un’immagine all’AEM Assets
Innanzitutto, carica un'immagine iniziale di esempio in AEM Assets da utilizzare per popolare l'immagine nel componente Byline.
-
Passare alla cartella Skateparks LA in AEM Assets: http://localhost:4502/assets.html/content/dam/wknd/en/magazine/la-skateparks.
-
Carica l'immagine principale per stacey-roswells.jpg nella cartella.
Creare il componente
Quindi, aggiungi il componente Byline a una pagina in AEM. Poiché il componente Byline viene aggiunto al gruppo di componenti WKND Sites Project - Content tramite la definizione ui.apps/src/main/content/jcr_root/apps/wknd/components/byline/.content.xml
, è automaticamente disponibile per qualsiasi Container il cui Policy consente il gruppo di componenti WKND Sites Project - Content. È quindi disponibile nel Contenitore di layout della pagina dell’articolo .
-
Passa all’articolo La Skatepark in: http://localhost:4502/editor.html/content/wknd/us/en/magazine/guide-la-skateparks.html
-
Dalla barra laterale a sinistra, trascina e rilascia un componente Byline in bottom del Contenitore di layout della pagina di articolo aperta.
-
Assicurati che la barra laterale a sinistra sia aperta e visibile e che sia selezionato Ricerca risorse**.
-
Seleziona il segnaposto del componente Byline, che a sua volta visualizza la barra delle azioni e tocca l'icona chiave inglese per aprire la finestra di dialogo.
-
Con la finestra di dialogo aperta e la prima scheda (Risorsa) attiva, apri la barra laterale a sinistra e, dal Finder risorse, trascina un’immagine nella zona di rilascio Immagine. Cerca "stacey" per trovare la bio-immagine Stacey Roswells fornita nel pacchetto WKND ui.content.
-
Dopo aver aggiunto un'immagine, fare clic sulla scheda Proprietà per immettere Nome e Occupazioni.
Quando inserisci le occupazioni, inseriscili in ordine alfabetico inverso in modo da verificare la logica di business alfabetizzata implementata nel modello Sling.
Tocca il pulsante Fine in basso a destra per salvare le modifiche.
Gli autori AEM configurano e creano i componenti tramite le finestre di dialogo. A questo punto, nello sviluppo del componente Byline sono incluse le finestre di dialogo per la raccolta dei dati, tuttavia non è stata ancora aggiunta la logica per il rendering del contenuto creato. Pertanto, viene visualizzato solo il segnaposto.
-
Dopo aver salvato la finestra di dialogo, passa a CRXDE Liti e controlla come il contenuto del componente viene memorizzato nel nodo del contenuto del componente byline nella pagina AEM.
Trovare il nodo di contenuto del componente Byline sotto la pagina Skate Parks di LA, ovvero
/content/wknd/us/en/magazine/guide-la-skateparks/jcr:content/root/container/container/byline
.Notare che i nomi di proprietà
name
,occupations
efileReference
sono archiviati nel nodo byline.Inoltre, tieni presente che
sling:resourceType
del nodo è impostato suwknd/components/content/byline
, che è ciò che associa questo nodo di contenuto all'implementazione del componente Byline.
Crea modello Sling Byline
Quindi, creiamo un modello Sling che funga da modello dati e racchiuda la logica di business per il componente Byline.
I modelli Sling sono POJO Java™ basati su annotazioni (Plain Old Java™ Objects) che semplificano la mappatura dei dati dalle variabili JCR a Java™ e forniscono efficienza durante lo sviluppo nel contesto AEM.
Esamina dipendenze Maven
Il modello Sling Byline si basa su diverse API Java™ fornite dall’AEM. Queste API sono rese disponibili tramite dependencies
elencato nel file POM del modulo core
. Il progetto utilizzato per questa esercitazione è stato creato per AEM as a Cloud Service. Tuttavia, è unico in quanto è compatibile con le versioni precedenti di AEM 6.5/6.4. Pertanto sono incluse entrambe le dipendenze per Cloud Service e AEM 6.x.
-
Apri il file
pom.xml
sotto<src>/aem-guides-wknd/core/pom.xml
. -
Trova la dipendenza per
aem-sdk-api
- Solo AEM as a Cloud Service<dependency> <groupId>com.adobe.aem</groupId> <artifactId>aem-sdk-api</artifactId> </dependency>
aem-sdk-api contiene tutte le API Java™ pubbliche esposte dall'AEM.
aem-sdk-api
viene utilizzato per impostazione predefinita durante la creazione di questo progetto. La versione viene mantenuta nel pom del reattore padre dalla radice del progetto inaem-guides-wknd/pom.xml
. -
Trova la dipendenza per
uber-jar
- Solo AEM 6.5/6.4... <dependency> <groupId>com.adobe.aem</groupId> <artifactId>uber-jar</artifactId> <classifier>apis</classifier> </dependency> ...
uber-jar
è incluso solo quando viene richiamato il profiloclassic
, ovveromvn clean install -PautoInstallSinglePackage -Pclassic
. Anche in questo caso, questo è specifico per questo progetto. In un progetto reale, generato dall'archetipo del progetto AEM,uber-jar
è l'impostazione predefinita se la versione dell'AEM specificata è 6.5 o 6.4.uber-jar contiene tutte le API Java™ pubbliche esposte da AEM 6.x. La versione viene mantenuta nel pom del reattore padre dalla radice del progetto
aem-guides-wknd/pom.xml
. -
Trovare la dipendenza per
core.wcm.components.core
:<!-- Core Component Dependency --> <dependency> <groupId>com.adobe.cq</groupId> <artifactId>core.wcm.components.core</artifactId> </dependency>
Si tratta delle API Java™ pubbliche complete esposte dai Componenti core AEM. I Componenti core AEM sono un progetto gestito al di fuori dell’AEM e hanno quindi un ciclo di rilascio separato. Per questo motivo, è una dipendenza che deve essere inclusa separatamente ed è non inclusa con
uber-jar
oaem-sdk-api
.Come uber-jar, la versione di questa dipendenza viene mantenuta nel file POM del reattore padre da
aem-guides-wknd/pom.xml
.Più avanti in questa esercitazione, la classe Immagine del componente core viene utilizzata per visualizzare l’immagine nel componente Byline. È necessario disporre della dipendenza dei Componenti core per generare e compilare il modello Sling.
Interfaccia Byline
Creare un'interfaccia Java™ pubblica per il nome del destinatario. Byline.java
definisce i metodi pubblici necessari per guidare lo script HTL byline.html
.
-
All'interno, il modulo
core
nella cartellacore/src/main/java/com/adobe/aem/guides/wknd/core/models
crea un file denominatoByline.java
-
Aggiorna
Byline.java
con i metodi seguenti:package com.adobe.aem.guides.wknd.core.models; import java.util.List; /** * Represents the Byline AEM Component for the WKND Site project. **/ public interface Byline { /*** * @return a string to display as the name. */ String getName(); /*** * Occupations are to be sorted alphabetically in a descending order. * * @return a list of occupations. */ List<String> getOccupations(); /*** * @return a boolean if the component has enough content to display. */ boolean isEmpty(); }
I primi due metodi espongono i valori per name e occupations per il componente Byline.
Il metodo
isEmpty()
viene utilizzato per determinare se il componente ha contenuto da riprodurre o se è in attesa di essere configurato.Nota che non esiste un metodo per l'immagine; questa versione verrà rivista più tardi.
-
I pacchetti Java™ che contengono classi Java™ pubbliche, in questo caso un modello Sling, devono essere sottoposti a controllo delle versioni utilizzando il file
package-info.java
del pacchetto.Poiché il pacchetto Java™
com.adobe.aem.guides.wknd.core.models
dell'origine WKND dichiara la versione di1.0.0
e vengono aggiunti un'interfaccia pubblica e metodi non interrompenti, è necessario aumentare la versione a1.1.0
. Aprire il file incore/src/main/java/com/adobe/aem/guides/wknd/core/models/package-info.java
e aggiornare@Version("1.0.0")
in@Version("2.1.0")
.@Version("2.1.0") package com.adobe.aem.guides.wknd.core.models; import org.osgi.annotation.versioning.Version;
Ogni volta che si apportano modifiche ai file di questo pacchetto, è necessario regolare semanticamente la versione del pacchetto 🔗. In caso contrario, bnd-baseline-maven-plugin del progetto Maven rileva una versione del pacchetto non valida e interrompe la generazione. Fortunatamente, in caso di errore, il plug-in Maven segnala la versione del pacchetto Java™ non valida e la versione corretta. Aggiornare la dichiarazione @Version("...")
nel pacchetto Java™ package-info.java
violato alla versione consigliata dal plug-in per la correzione.
Implementazione in linea
BylineImpl.java
è l'implementazione del modello Sling che implementa l'interfaccia Byline.java
definita in precedenza. Il codice completo per BylineImpl.java
si trova in fondo a questa sezione.
-
Creare una cartella denominata
impl
sottocore/src/main/java/com/adobe/aem/guides/core/models
. -
Nella cartella
impl
, creare un fileBylineImpl.java
. -
Apri
BylineImpl.java
. Specificare che implementa l'interfacciaByline
. Utilizzare le funzionalità di completamento automatico dell'IDE o aggiornare manualmente il file per includere i metodi necessari per implementare l'interfacciaByline
:package com.adobe.aem.guides.wknd.core.models.impl; import java.util.List; import com.adobe.aem.guides.wknd.core.models.Byline; public class BylineImpl implements Byline { @Override public String getName() { // TODO Auto-generated method stub return null; } @Override public List<String> getOccupations() { // TODO Auto-generated method stub return null; } @Override public boolean isEmpty() { // TODO Auto-generated method stub return false; } }
-
Aggiungi le annotazioni del modello Sling aggiornando
BylineImpl.java
con le seguenti annotazioni a livello di classe. Questa annotazione@Model(..)
è ciò che trasforma la classe in un modello Sling.import org.apache.sling.api.SlingHttpServletRequest; import org.apache.sling.models.annotations.Model; import org.apache.sling.models.annotations.DefaultInjectionStrategy; ... @Model( adaptables = {SlingHttpServletRequest.class}, adapters = {Byline.class}, resourceType = {BylineImpl.RESOURCE_TYPE}, defaultInjectionStrategy = DefaultInjectionStrategy.OPTIONAL ) public class BylineImpl implements Byline { protected static final String RESOURCE_TYPE = "wknd/components/byline"; ... }
Esaminiamo questa annotazione e i relativi parametri:
- L'annotazione
@Model
registra BylineImpl come modello Sling quando viene distribuito in AEM. - Il parametro
adaptables
specifica che questo modello può essere adattato dalla richiesta. - Il parametro
adapters
consente la registrazione della classe di implementazione nell'interfaccia Byline. Questo consente allo script HTL di chiamare il modello Sling tramite l’interfaccia (anziché direttamente l’implementazione). Ulteriori dettagli sulle schede sono disponibili qui. resourceType
punta al tipo di risorsa del componente Byline (creato in precedenza) e aiuta a risolvere il modello corretto in presenza di più implementazioni. Ulteriori dettagli sull'associazione di una classe modello a un tipo di risorsa sono disponibili qui.
- L'annotazione
Implementazione dei metodi del modello Sling
getName()
Il primo metodo implementato è getName()
. Restituisce semplicemente il valore memorizzato nel nodo di contenuto JCR della riga di comando sotto la proprietà name
.
Per questo motivo, l'annotazione del modello Sling @ValueMapValue
viene utilizzata per inserire il valore in un campo Java™ utilizzando ValueMap della risorsa della richiesta.
import org.apache.sling.models.annotations.injectorspecific.ValueMapValue;
public class BylineImpl implements Byline {
...
@ValueMapValue
private String name;
...
@Override
public String getName() {
return name;
}
...
}
Poiché la proprietà JCR condivide il nome come campo Java™ (entrambi sono "name"), @ValueMapValue
risolve automaticamente questa associazione e inserisce il valore della proprietà nel campo Java™.
getOccupations()
Il metodo successivo da implementare è getOccupations()
. Questo metodo carica le occupazioni memorizzate nella proprietà JCR occupations
e restituisce un insieme ordinato (alfabeticamente) di tali occupazioni.
Utilizzando la stessa tecnica illustrata in getName()
, il valore della proprietà può essere inserito nel campo del modello Sling.
Una volta che i valori delle proprietà JCR sono disponibili nel modello Sling tramite il campo Java™ inserito occupations
, è possibile applicare la logica di business di ordinamento nel metodo getOccupations()
.
import java.util.ArrayList;
import java.util.Collections;
...
public class BylineImpl implements Byline {
...
@ValueMapValue
private List<String> occupations;
...
@Override
public List<String> getOccupations() {
if (occupations != null) {
Collections.sort(occupations);
return new ArrayList<String>(occupations);
} else {
return Collections.emptyList();
}
}
...
}
...
isEmpty()
L'ultimo metodo pubblico è isEmpty()
che determina quando il componente deve considerarsi "creato abbastanza" per il rendering.
Per questo componente, il requisito aziendale è costituito da tutti e tre i campi, name, image and occupations
deve essere compilato prima che sia possibile eseguire il rendering del componente.
import org.apache.commons.lang3.StringUtils;
...
public class BylineImpl implements Byline {
...
@Override
public boolean isEmpty() {
if (StringUtils.isBlank(name)) {
// Name is missing, but required
return true;
} else if (occupations == null || occupations.isEmpty()) {
// At least one occupation is required
return true;
} else if (/* image is not null, logic to be determined */) {
// A valid image is required
return true;
} else {
// Everything is populated, so this component is not considered empty
return false;
}
}
...
}
Affrontare il "problema dell'immagine"
La verifica del nome e delle condizioni di occupazione è banale e Apache Commons Lang3 fornisce la classe StringUtils utile. Tuttavia, non è chiaro come la presenza dell'immagine possa essere convalidata poiché il componente core Immagine viene utilizzato per far emergere l'immagine.
Esistono due modi per affrontare questo problema:
Verifica se la proprietà JCR fileReference
viene risolta in una risorsa. OR Converti questa risorsa in un modello Sling di immagine Componente core e assicurati che il metodo getSrc()
non sia vuoto.
Usiamo l'approccio second. Il primo approccio è probabilmente sufficiente, ma in questo tutorial viene utilizzato per consentirci di esplorare altre funzioni dei modelli Sling.
-
Crea un metodo privato per ottenere l'immagine. Questo metodo è lasciato privato perché non è necessario esporre l'oggetto Image nell'HTL stesso ed è utilizzato solo per guidare
isEmpty().
Aggiungi il seguente metodo privato per
getImage()
:import com.adobe.cq.wcm.core.components.models.Image; ... private Image getImage() { Image image = null; // Figure out how to populate the image variable! return image; }
Come indicato in precedenza, esistono altri due approcci per ottenere il modello Sling per immagini:
Il primo utilizza l'annotazione
@Self
, per adattare automaticamente la richiesta corrente alImage.class
del componente coreIl secondo utilizza il servizio OSGi Apache Sling ModelFactory, che è un servizio utile e ci aiuta a creare modelli Sling di altri tipi nel codice Java™.
Usiamo il secondo approccio.
NOTEIn un'implementazione reale, l'approccio "Uno", che utilizza@Self
, è da preferirsi in quanto è la soluzione più semplice ed elegante. In questo tutorial viene utilizzato il secondo approccio, in quanto richiede di esplorare più facet di modelli Sling che sono utili è componenti più complessi!Poiché i modelli Sling sono Java™ POJO e non OSGi Services, non è possibile utilizzare le consuete annotazioni di iniezione OSGi
@Reference
2}, ma i modelli Sling forniscono un'annotazione speciale @OSGiServiceche offre funzionalità simili. -
Aggiorna
BylineImpl.java
per includere l'annotazioneOSGiService
per inserireModelFactory
:import org.apache.sling.models.factory.ModelFactory; import org.apache.sling.models.annotations.injectorspecific.OSGiService; ... public class BylineImpl implements Byline { ... @OSGiService private ModelFactory modelFactory; }
Con
ModelFactory
disponibile, è possibile creare un modello Sling immagine componente core utilizzando:modelFactory.getModelFromWrappedRequest(SlingHttpServletRequest request, Resource resource, java.lang.Class<T> targetClass)
Tuttavia, questo metodo richiede sia una richiesta che una risorsa, non ancora disponibili nel modello Sling. Per ottenere questi risultati, vengono utilizzate più annotazioni del modello Sling.
Per ottenere la richiesta corrente è possibile utilizzare l'annotazione @Self per inserire
adaptable
(definito in@Model(..)
comeSlingHttpServletRequest.class
, in un campo di classe Java™. -
Aggiungi l'annotazione @Self per ottenere la richiesta SlingHttpServletRequest:
import org.apache.sling.models.annotations.injectorspecific.Self; ... @Self private SlingHttpServletRequest request;
Ricorda che l’utilizzo di
@Self Image image
per inserire il modello Sling dell’immagine del componente core era un’opzione di cui sopra: l’annotazione@Self
tenta di inserire l’oggetto adattabile (in questo caso SlingHttpServletRequest) e di adattarsi al tipo di campo dell’annotazione. Poiché il modello Sling dell’immagine del componente core è adattabile dagli oggetti SlingHttpServletRequest, questo avrebbe funzionato ed è meno codice rispetto all’approcciomodelFactory
più esplorativo.Ora vengono inserite le variabili necessarie per creare un’istanza del modello Immagine tramite l’API ModelFactory. Utilizziamo l'annotazione @PostConstruct del modello Sling per ottenere questo oggetto dopo le istanze del modello Sling.
@PostConstruct
è incredibilmente utile e agisce in una capacità simile come costruttore, tuttavia, viene richiamato dopo la creazione dell'istanza della classe e l'inserimento di tutti i campi Java™ con annotazioni. Mentre altre annotazioni del modello Sling annotano campi di classe Java™ (variabili),@PostConstruct
annota un metodo con parametro void, zero, in genere denominatoinit()
(ma può essere denominato qualsiasi cosa). -
Aggiungi metodo @PostConstruct:
import javax.annotation.PostConstruct; ... public class BylineImpl implements Byline { ... private Image image; @PostConstruct private void init() { image = modelFactory.getModelFromWrappedRequest(request, request.getResource(), Image.class); } ... }
Ricorda che i modelli Sling sono NOT servizi OSGi, pertanto è sicuro mantenere lo stato della classe. Spesso
@PostConstruct
deriva e imposta lo stato della classe del modello Sling per un utilizzo successivo, in modo simile a quello di un costruttore semplice.Se il metodo
@PostConstruct
genera un'eccezione, il modello Sling non viene creato come istanza ed è nullo. -
È ora possibile aggiornare getImage() per restituire semplicemente l'oggetto immagine.
/** * @return the Image Sling Model of this resource, or null if the resource cannot create a valid Image Sling Model. */ private Image getImage() { return image; }
-
Torniamo a
isEmpty()
e terminiamo l'implementazione:@Override public boolean isEmpty() { final Image componentImage = getImage(); if (StringUtils.isBlank(name)) { // Name is missing, but required return true; } else if (occupations == null || occupations.isEmpty()) { // At least one occupation is required return true; } else if (componentImage == null || StringUtils.isBlank(componentImage.getSrc())) { // A valid image is required return true; } else { // Everything is populated, so this component is not considered empty return false; } }
Nota: più chiamate a
getImage()
non sono problematiche in quanto restituisce la variabile di classeimage
inizializzata e non richiamamodelFactory.getModelFromWrappedRequest(...)
, il che non è eccessivamente costoso, ma vale la pena evitare di chiamare inutilmente. -
L'aspetto finale di
BylineImpl.java
dovrebbe essere il seguente:package com.adobe.aem.guides.wknd.core.models.impl; import java.util.ArrayList; import java.util.Collections; import java.util.List; import javax.annotation.PostConstruct; import org.apache.commons.lang3.StringUtils; import org.apache.sling.api.SlingHttpServletRequest; import org.apache.sling.models.annotations.DefaultInjectionStrategy; import org.apache.sling.models.annotations.Model; import org.apache.sling.models.annotations.injectorspecific.OSGiService; import org.apache.sling.models.annotations.injectorspecific.Self; import org.apache.sling.models.annotations.injectorspecific.ValueMapValue; import org.apache.sling.models.factory.ModelFactory; import com.adobe.aem.guides.wknd.core.models.Byline; import com.adobe.cq.wcm.core.components.models.Image; @Model( adaptables = {SlingHttpServletRequest.class}, adapters = {Byline.class}, resourceType = {BylineImpl.RESOURCE_TYPE}, defaultInjectionStrategy = DefaultInjectionStrategy.OPTIONAL ) public class BylineImpl implements Byline { protected static final String RESOURCE_TYPE = "wknd/components/byline"; @Self private SlingHttpServletRequest request; @OSGiService private ModelFactory modelFactory; @ValueMapValue private String name; @ValueMapValue private List<String> occupations; private Image image; /** * @PostConstruct is immediately called after the class has been initialized * but BEFORE any of the other public methods. * It is a good method to initialize variables that is used by methods in the rest of the model * */ @PostConstruct private void init() { // set the image object image = modelFactory.getModelFromWrappedRequest(request, request.getResource(), Image.class); } @Override public String getName() { return name; } @Override public List<String> getOccupations() { if (occupations != null) { Collections.sort(occupations); return new ArrayList<String>(occupations); } else { return Collections.emptyList(); } } @Override public boolean isEmpty() { final Image componentImage = getImage(); if (StringUtils.isBlank(name)) { // Name is missing, but required return true; } else if (occupations == null || occupations.isEmpty()) { // At least one occupation is required return true; } else if (componentImage == null || StringUtils.isBlank(componentImage.getSrc())) { // A valid image is required return true; } else { // Everything is populated, so this component is not considered empty return false; } } /** * @return the Image Sling Model of this resource, or null if the resource cannot create a valid Image Sling Model. */ private Image getImage() { return image; } }
Byline HTL
Nel modulo ui.apps
, aprire /apps/wknd/components/byline/byline.html
creato nella configurazione precedente del componente AEM.
<div data-sly-use.placeholderTemplate="core/wcm/components/commons/v1/templates.html">
</div>
<sly data-sly-call="${placeholderTemplate.placeholder @ isEmpty=false}"></sly>
Esaminiamo cosa fa finora questo script HTL:
-
placeholderTemplate
punta al segnaposto dei Componenti core, che viene visualizzato quando il componente non è completamente configurato. Viene eseguito il rendering nell'Editor pagina di AEM Sites come una casella con il titolo del componente, come definito in precedenza nella proprietàjcr:title
dicq:Component
. -
data-sly-call="${placeholderTemplate.placeholder @ isEmpty=false}
caricaplaceholderTemplate
definito in precedenza e passa un valore booleano (attualmente hardcoded infalse
) nel modello segnaposto. QuandoisEmpty
è true, il modello segnaposto esegue il rendering della casella grigia, altrimenti non esegue il rendering.
Aggiorna il codice HTL della riga di comando
-
Aggiorna byline.html con la seguente struttura HTML scheletrica:
<div data-sly-use.placeholderTemplate="core/wcm/components/commons/v1/templates.html" class="cmp-byline"> <div class="cmp-byline__image"> <!--/* Include the Core Components Image Component */--> </div> <h2 class="cmp-byline__name"><!--/* Include the name */--></h2> <p class="cmp-byline__occupations"><!--/* Include the occupations */--></p> </div> <sly data-sly-call="${placeholderTemplate.placeholder @ isEmpty=true}"></sly>
Le classi CSS seguono la convenzione di denominazione BEM. Anche se l’utilizzo delle convenzioni BEM non è obbligatorio, BEM è consigliato in quanto viene utilizzato nelle classi CSS dei Componenti core e in genere si traduce in regole CSS chiare e leggibili.
Creazione di istanze di oggetti del modello Sling in HTL
L'istruzione Use Block viene utilizzata per creare istanze di oggetti modello Sling nello script HTL e assegnarla a una variabile HTL.
data-sly-use.byline="com.adobe.aem.guides.wknd.models.Byline"
utilizza l'interfaccia Byline (com.adobe.aem.guides.wknd.models.Byline) implementata da BylineImpl e adatta ad essa la SlingHttpServletRequest corrente. Il risultato viene memorizzato in una riga per il nome della variabile HTL ( data-sly-use.<variable-name>
).
-
Aggiorna l'esterno
div
in modo che faccia riferimento al modello Sling Byline tramite la relativa interfaccia pubblica:<div data-sly-use.byline="com.adobe.aem.guides.wknd.core.models.Byline" data-sly-use.placeholderTemplate="core/wcm/components/commons/v1/templates.html" class="cmp-byline"> ... </div>
Accesso ai metodi del modello Sling
HTL prende in prestito da JSTL e utilizza lo stesso accorciamento dei nomi dei metodi getter Java™.
Ad esempio, richiamare il metodo getName()
del modello Sling Byline può essere ridotto a byline.name
, analogamente invece di byline.isEmpty
, può essere ridotto a byline.empty
. Anche l'utilizzo dei nomi di metodo completi, byline.getName
o byline.isEmpty
, funziona. Nota che ()
non vengono mai utilizzati per richiamare metodi in HTL (simile a JSTL).
I metodi Java™ che richiedono un parametro non possono essere utilizzati in HTL. Questo è progettato per mantenere la logica in HTL semplice.
-
È possibile aggiungere il nome della riga di comando al componente richiamando il metodo
getName()
nel modello Sling Byline oppure in HTL:${byline.name}
.Aggiorna il tag
h2
:<h2 class="cmp-byline__name">${byline.name}</h2>
Utilizzo delle opzioni delle espressioni HTL
Opzioni espressioni HTL fungono da modificatori del contenuto in HTL e vanno dalla formattazione della data alla traduzione i18n. Le espressioni possono essere utilizzate anche per unire elenchi o array di valori, necessari per visualizzare le occupazioni in un formato delimitato da virgole.
Le espressioni vengono aggiunte tramite l’operatore @
nell’espressione HTL.
-
Per unire l’elenco delle occupazioni con ", ", viene utilizzato il seguente codice:
<p class="cmp-byline__occupations">${byline.occupations @ join=', '}</p>
Visualizzazione condizionale del segnaposto
La maggior parte degli script HTL per i componenti AEM utilizza il paradigma placeholder per fornire un segnale visivo agli autori che indica che un componente è creato in modo errato e non viene visualizzato in AEM Publish. La convenzione per guidare questa decisione consiste nell'implementare un metodo sul modello Sling di supporto del componente, in questo caso: Byline.isEmpty()
.
Il metodo isEmpty()
viene richiamato nel modello Sling Byline e il risultato (o piuttosto negativo, tramite l'operatore !
) viene salvato in una variabile HTL denominata hasContent
:
-
Aggiornare l'esterno
div
per salvare una variabile HTL denominatahasContent
:<div data-sly-use.byline="com.adobe.aem.guides.wknd.core.models.Byline" data-sly-use.placeholderTemplate="core/wcm/components/commons/v1/templates.html" data-sly-test.hasContent="${!byline.empty}" class="cmp-byline"> ... </div>
Nota l'utilizzo di
data-sly-test
, il blocco HTLtest
è chiave, imposta una variabile HTL ed esegue/non esegue il rendering dell'elemento HTML su cui si trova. Si basa sul risultato della valutazione dell’espressione HTL. Se "true", viene eseguito il rendering dell’elemento HTML, altrimenti non viene eseguito il rendering.È ora possibile riutilizzare questa variabile HTL
hasContent
per mostrare/nascondere il segnaposto in modo condizionale. -
Aggiorna la chiamata condizionale a
placeholderTemplate
nella parte inferiore del file con quanto segue:<sly data-sly-call="${placeholderTemplate.placeholder @ isEmpty=!hasContent}"></sly>
Visualizzare l’immagine utilizzando i componenti core
Lo script HTL per byline.html
è quasi completato e manca solo l'immagine.
Poiché sling:resourceSuperType
punta al componente Immagine del componente core per creare l'immagine, è possibile utilizzare il componente Immagine del componente core per eseguire il rendering dell'immagine.
A questo scopo, includiamo la risorsa byline corrente, ma forziamo il tipo di risorsa del componente immagine del componente core, utilizzando il tipo di risorsa core/wcm/components/image/v2/image
. Si tratta di un modello potente per il riutilizzo di componenti. Per questo, viene utilizzato il blocco data-sly-resource
di HTL.
-
Sostituire
div
con una classe dicmp-byline__image
con quanto segue:<div class="cmp-byline__image" data-sly-resource="${ '.' @ resourceType = 'core/wcm/components/image/v2/image' }"></div>
data-sly-resource
, include la risorsa corrente tramite il percorso relativo'.'
e forza l'inclusione della risorsa corrente (o della risorsa di contenuto byline) con il tipo di risorsacore/wcm/components/image/v2/image
.Il tipo di risorsa Componente core viene utilizzato direttamente e non tramite un proxy, perché si tratta di un utilizzo in-script e non viene mai mantenuto nel contenuto.
-
Completato
byline.html
di seguito:<!--/* byline.html */--> <div data-sly-use.byline="com.adobe.aem.guides.wknd.core.models.Byline" data-sly-use.placeholderTemplate="core/wcm/components/commons/v1/templates.html" data-sly-test.hasContent="${!byline.empty}" class="cmp-byline"> <div class="cmp-byline__image" data-sly-resource="${ '.' @ resourceType = 'core/wcm/components/image/v2/image' }"> </div> <h2 class="cmp-byline__name">${byline.name}</h2> <p class="cmp-byline__occupations">${byline.occupations @ join=', '}</p> </div> <sly data-sly-call="${placeholderTemplate.placeholder @ isEmpty=!hasContent}"></sly>
-
Distribuire la base di codice in un'istanza AEM locale. Poiché sono state apportate modifiche a
core
eui.apps
, è necessario distribuire entrambi i moduli.$ cd aem-guides-wknd/ui.apps $ mvn clean install -PautoInstallPackage
$ cd ../core $ mvn clean install -PautoInstallBundle
Per distribuire a AEM 6.5/6.4, richiamare il profilo
classic
:$ cd ../core $ mvn clean install -PautoInstallBundle -Pclassic
CAUTIONÈ inoltre possibile creare l'intero progetto dalla directory principale utilizzando il profilo MavenautoInstallSinglePackage
, ma questo potrebbe sovrascrivere le modifiche al contenuto nella pagina. Questo perchéui.content/src/main/content/META-INF/vault/filter.xml
è stato modificato per consentire al codice iniziale dell'esercitazione di sovrascrivere in modo chiaro il contenuto AEM esistente. In uno scenario reale, questo non è un problema.
Revisione del componente Byline non formattato
-
Dopo aver distribuito l'aggiornamento, passare alla pagina Ultimate Guide to LA Skateparks o a qualsiasi altro componente Byline aggiunto in precedenza nel capitolo.
-
Le immagini, nomi e occupazioni sono ora visualizzate e non dispongono di uno stile, ma è presente un componente Byline funzionante.
Revisione della registrazione del modello Sling
La visualizzazione dello stato dei modelli Sling della console Web AEM visualizza tutti i modelli Sling registrati nell'AEM. Esaminando questo elenco è possibile convalidare e riconoscere il modello Sling Byline come installato.
Se BylineImpl non è visualizzato in questo elenco, è probabile che si sia verificato un problema con le annotazioni del modello Sling o che il modello non sia stato aggiunto al pacchetto corretto (com.adobe.aem.guides.wknd.core.models
) nel progetto di base.
http://localhost:4502/system/console/status-slingmodels
Stili linea di base
Per allineare il componente Byline con il design creativo fornito, aggiungiamo uno stile. Ciò si ottiene utilizzando il file SCSS e aggiornando il file nel modulo ui.frontend.
Aggiungi uno stile predefinito
Aggiungi gli stili predefiniti per il componente Byline.
-
Torna all'IDE e al progetto ui.frontend in
/src/main/webpack/components
: -
Creare un file denominato
_byline.scss
. -
Aggiungere le implementazioni di Byline CSS (scritte come SCSS) in
_byline.scss
:.cmp-byline { $imageSize: 60px; .cmp-byline__image { float: left; /* This class targets a Core Component Image CSS class */ .cmp-image__image { width: $imageSize; height: $imageSize; border-radius: $imageSize / 2; object-fit: cover; } } .cmp-byline__name { font-size: $font-size-medium; font-family: $font-family-serif; padding-top: 0.5rem; margin-left: $imageSize + 25px; margin-bottom: .25rem; margin-top:0rem; } .cmp-byline__occupations { margin-left: $imageSize + 25px; color: $gray; font-size: $font-size-xsmall; text-transform: uppercase; } }
-
Aprire un terminale e passare al modulo
ui.frontend
. -
Avvia il processo
watch
con il seguente comando npm:$ cd ui.frontend/ $ npm run watch
-
Torna al browser e passa all'articolo LA SkateParks. Dovresti visualizzare gli stili aggiornati del componente.
TIPPotrebbe essere necessario cancellare la cache del browser per garantire che non vengano forniti CSS non aggiornati e aggiornare la pagina con il componente Byline per ottenere lo stile completo.
Congratulazioni.
Congratulazioni, hai creato un componente personalizzato partendo da zero utilizzando Adobe Experience Manager.
Passaggi successivi
Continua a scoprire lo sviluppo dei componenti AEM esplorando come scrivere test JUnit per il codice Java™ Byline per garantire che tutto sia sviluppato correttamente e che la logica di business implementata sia corretta e completa.
Visualizza il codice finito in GitHub oppure rivedi e distribuisci il codice localmente in nel ramo Git tutorial/custom-component-solution
.
- Clona l'archivio github.com/adobe/aem-guides-wknd.
- Controlla il ramo
tutorial/custom-component-solution