Last active
July 22, 2025 02:21
-
-
Save prince-neres/090162aa3da4d39c3f3d09a93c74cc5d to your computer and use it in GitHub Desktop.
Liferay card fragment
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| { | |
| "fieldSets": [ | |
| { | |
| "fields": [ | |
| { | |
| "name": "horizontalCard", | |
| "label": "Card Horizonal", | |
| "type": "checkbox", | |
| "defaultValue": true | |
| }, | |
| { | |
| "name": "hideImage", | |
| "label": "Esconder Imagem", | |
| "type": "checkbox", | |
| "defaultValue": false | |
| }, | |
| { | |
| "name": "hideCategories", | |
| "label": "Esconder Categorias", | |
| "type": "checkbox", | |
| "defaultValue": false | |
| }, | |
| { | |
| "name": "categoryBadgeBgColor", | |
| "label": "Cor de Fundo da Categoria", | |
| "type": "colorPicker", | |
| "defaultValue": "var(--color-state-info-lighten-2)" | |
| }, | |
| { | |
| "name": "categoryBadgeTextColor", | |
| "label": "Cor do Texto da Categoria", | |
| "type": "colorPicker", | |
| "defaultValue": "var(--info)" | |
| }, | |
| { | |
| "name": "hideOverlineText", | |
| "label": "Esconder Sobrelinha", | |
| "type": "checkbox", | |
| "defaultValue": true | |
| }, | |
| { | |
| "name": "hideText", | |
| "label": "Esconder Texto", | |
| "type": "checkbox", | |
| "defaultValue": false | |
| }, | |
| { | |
| "name": "hideButton", | |
| "label": "Esconder Botão", | |
| "type": "checkbox", | |
| "defaultValue": true | |
| }, | |
| { | |
| "name": "hideShareIcon", | |
| "label": "Esconder Ícone de Compartilhar", | |
| "type": "checkbox", | |
| "defaultValue": true | |
| }, | |
| { | |
| "name": "hideDateText", | |
| "label": "Esconder Texto de Data", | |
| "type": "checkbox", | |
| "defaultValue": false | |
| }, | |
| { | |
| "name": "dateTextFormat", | |
| "label": "Formato do Texto da Data", | |
| "type": "select", | |
| "dataType": "string", | |
| "typeOptions": { | |
| "validValues": [ | |
| { | |
| "label": "Atualizado pela ultima vez 10 Horas.", | |
| "value": "news" | |
| }, | |
| { | |
| "label": "Postado em 1 de Janeiro de 2024.", | |
| "value": "events" | |
| }, | |
| { | |
| "label": "01/01/2024", | |
| "value": "default" | |
| } | |
| ] | |
| }, | |
| "defaultValue": "default" | |
| }, | |
| { | |
| "name": "titleLineClamp", | |
| "label": "Limite de Linhas do Título", | |
| "type": "select", | |
| "dataType": "int", | |
| "typeOptions": { | |
| "validValues": [ | |
| {"value": "1"}, | |
| {"value": "2"}, | |
| {"value": "3"}, | |
| {"value": "4"}, | |
| {"value": "5"} | |
| ] | |
| }, | |
| "defaultValue": "3" | |
| }, | |
| { | |
| "name": "textLineClamp", | |
| "label": "Limite de Linhas do Texto", | |
| "type": "select", | |
| "dataType": "int", | |
| "typeOptions": { | |
| "validValues": [ | |
| {"value": "1"}, | |
| {"value": "2"}, | |
| {"value": "3"}, | |
| {"value": "4"}, | |
| {"value": "5"} | |
| ] | |
| }, | |
| "defaultValue": "3" | |
| } | |
| ] | |
| } | |
| ] | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| { | |
| "configurationPath": "configuration.json", | |
| "jsPath": "index.js", | |
| "htmlPath": "index.html", | |
| "cssPath": "index.css", | |
| "icon": "code", | |
| "name": "Card Customizado", | |
| "type": "component" | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| .lfr-layout-structure-item-collection .container-fluid:has(.custom-card) { | |
| overflow: visible !important; | |
| } | |
| .custom-card { | |
| border: none; | |
| box-shadow: 0 2px 6px 0 #00000038; | |
| border-radius: var(--border-radius-lg); | |
| overflow: hidden; | |
| } | |
| .custom-card .card-img { | |
| border-radius: 0; | |
| object-fit: cover; | |
| object-position: center; | |
| } | |
| .custom-card .card-img-left { | |
| height: 100%; | |
| object-fit: cover; | |
| } | |
| .custom-card .card-img-top { | |
| height: 9.375rem; | |
| } | |
| .card-horizontal .card-body { | |
| padding: 0.938rem 1.563rem; | |
| } | |
| .card-vertical .card-body { | |
| padding: 1.563rem 0.938rem; | |
| } | |
| .custom-card .card-text-overline p { | |
| font-weight: bold; | |
| font-size: var(--text-small-caps-font-size); | |
| margin-bottom: var(--spacer-1); | |
| } | |
| .custom-card .card-title a { | |
| text-underline-offset: auto; | |
| } | |
| .custom-card .card-title { | |
| line-height: 1.5rem; | |
| } | |
| .custom-card .card-title { | |
| display: -webkit-box; | |
| -webkit-box-orient: vertical; | |
| overflow: hidden; | |
| } | |
| .custom-card .card-text { | |
| display: -webkit-box; | |
| -webkit-box-orient: vertical; | |
| overflow: hidden; | |
| } | |
| .custom-card .card-text a { | |
| color: var(--color-neutral-8) !important; | |
| text-decoration: none !important; | |
| } | |
| .custom-card .card-categories { | |
| max-height: 3.5rem; | |
| overflow: hidden; | |
| } | |
| .custom-card .card-categories span { | |
| text-wrap: nowrap; | |
| } | |
| .custom-card .card-btn { | |
| color: var(--color-brand-primary-darken-5); | |
| font-weight: bold; | |
| border: 2px solid var(--color-brand-primary-darken-5); | |
| border-radius: var(--border-radius); | |
| text-decoration: none; | |
| } | |
| .custom-card .card-share-link { | |
| cursor: pointer; | |
| } | |
| .custom-card .card-share-link svg { | |
| fill: var(--color-neutral-4); | |
| height: 1.5rem; | |
| } | |
| .custom-card .card-text, | |
| .custom-card .card-date { | |
| font-size: var(--text-paragraph-sm-font-size); | |
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| [#assign assetEntryLocalService = serviceLocator.findService("com.liferay.asset.kernel.service.AssetEntryLocalService") /] | |
| [#assign groupLocalService = serviceLocator.findService("com.liferay.portal.kernel.service.GroupLocalService") /] | |
| [#assign journalArticleLocalService = serviceLocator.findService("com.liferay.journal.service.JournalArticleLocalService") /] | |
| [#assign | |
| groupFriendlyURL = groupLocalService.getGroup(themeDisplay.getSiteGroupId()).getFriendlyURL() | |
| objectItemReference = (request.getAttribute("INFO_ITEM_REFERENCE"))!"" | |
| entryCategories = ["Categoria 1", "Categoria 2"] | |
| cardDateMessage = "" | |
| cardOrientationSlug = configuration.horizontalCard?then('horizontal', 'vertical') | |
| cardDateTextFormat = configuration.dateTextFormat | |
| entryModifiedDate = "" | |
| entryPublishDate = "" | |
| title = "" | |
| displayPageUrl = "" | |
| /] | |
| [#if objectItemReference != ""] | |
| [#assign | |
| classPK = objectItemReference.getInfoItemIdentifier().getClassPK()!0 | |
| className = objectItemReference.getClassName()!"" | |
| entry = assetEntryLocalService.getEntry(className, classPK) | |
| entryCategories = entry.getCategories() | |
| entryModifiedDate = entry.getModifiedDate() | |
| entryPublishDate = entry.getPublishDate() | |
| journalArticle = journalArticleLocalService.fetchLatestArticle(classPK) | |
| assetRenderer = entry.getAssetRenderer()!"" | |
| displayPageUrl = "/web" + groupFriendlyURL + "/w/" + assetRenderer.getUrlTitle() | |
| title = entry.getTitle(locale) | |
| /] | |
| [/#if] | |
| <div class="custom-card card card-${cardOrientationSlug}"> | |
| [#if configuration.horizontalCard] | |
| <div class="row no-gutters"> | |
| <div class="${configuration.hideImage?then('', 'col-4')}"> | |
| [/#if] | |
| [#if !configuration.hideImage] | |
| <img | |
| src="" | |
| class="card-img ${configuration.horizontalCard?then('card-img-left', 'card-img-top')}" | |
| data-lfr-editable-type="image" | |
| data-lfr-editable-id="Imagem" | |
| alt="..." | |
| /> | |
| [/#if] | |
| [#if configuration.horizontalCard] | |
| </div> | |
| <div class="${configuration.hideImage?then('', 'col-8')}"> | |
| [/#if] | |
| <div class="card-body"> | |
| [#if entryCategories?size > 0 && !configuration.hideCategories] | |
| <div class="card-categories d-flex flex-wrap"> | |
| [#assign badgeBgColor = configuration.categoryBadgeBgColor /] | |
| [#assign badgeTextColor = configuration.categoryBadgeTextColor /] | |
| [#list 0..entryCategories?size-1 as category] | |
| <span | |
| class="rounded font-weight-bold p-2 mb-3 ${(category==0)?then('mr-3', '')}" | |
| style="background-color: ${badgeBgColor}; color: ${badgeTextColor};" | |
| > | |
| [#if entryCategories[category]?is_hash] | |
| ${entryCategories[category].getTitle(locale)} | |
| [#else] | |
| ${entryCategories[category]} | |
| [/#if] | |
| </span> | |
| [#if category == 2] | |
| [#break] | |
| [/#if] | |
| [/#list] | |
| </div> | |
| [/#if] | |
| [#if !configuration.hideOverlineText] | |
| <div class="card-text-overline"> | |
| <p | |
| data-lfr-editable-type="text" | |
| data-lfr-editable-id="Sobrelinha" | |
| > | |
| SOBRELINHA | |
| </p> | |
| </div> | |
| [/#if] | |
| <h3 | |
| class="card-title mb-3" | |
| data-lfr-editable-type="text" | |
| data-lfr-editable-id="Título" | |
| style="-webkit-line-clamp: ${configuration.titleLineClamp};" | |
| > | |
| Título | |
| </h3> | |
| [#if !configuration.hideText] | |
| <p | |
| class="card-text" | |
| data-lfr-editable-type="text" | |
| data-lfr-editable-id="Texto" | |
| style="-webkit-line-clamp: ${configuration.textLineClamp};" | |
| > | |
| ${configuration.horizontalCard?then( | |
| "Este é um cartão mais largo com texto de apoio abaixo como uma introdução natural para conteúdo adicional. Este conteúdo é um pouco mais longo.", | |
| "Um texto de exemplo rápido para desenvolver o título do cartão e compor a maior parte do conteúdo do cartão." | |
| )} | |
| </p> | |
| [/#if] | |
| [#if (!configuration.hideButton || !configuration.hideShareIcon)] | |
| <div class="card-actions w-100 d-flex justify-content-between align-items-center mb-3"> | |
| [#if !configuration.hideButton] | |
| <a | |
| class="card-btn btn btn-link px-3 py-2" | |
| data-lfr-editable-type="link" | |
| data-lfr-editable-id="Botão" | |
| > | |
| Botão | |
| </a> | |
| [/#if] | |
| [#if !configuration.hideShareIcon] | |
| <span | |
| class="card-share-link" | |
| data-share-title="${title}" | |
| data-share-url="${displayPageUrl}" | |
| > | |
| [@clay["icon"] symbol="share" /] | |
| </span> | |
| [/#if] | |
| </div> | |
| [/#if] | |
| [#if !configuration.hideDateText && entryPublishDate?is_date] | |
| <p | |
| class="card-date mb-0" | |
| [#if cardDateTextFormat == "default"] | |
| data-lfr-editable-type="date-time" | |
| data-lfr-editable-id="Texto de Data" | |
| [/#if] | |
| > | |
| [#if cardDateTextFormat == "default"] | |
| ${entryPublishDate?date?string("dd/MM/yyyy")} | |
| [#elseif cardDateTextFormat == "news"] | |
| ${getLastUpdateTime(entryModifiedDate)} | |
| [#elseif cardDateTextFormat == "events"] | |
| ${getPostDate(entryPublishDate)} | |
| [/#if] | |
| </p> | |
| [/#if] | |
| </div> | |
| [#if configuration.horizontalCard] | |
| </div> | |
| [/#if] | |
| </div> | |
| [#function getPostDate previous] | |
| [#assign | |
| monthKey = previous?date?string("MMMM") | |
| month = languageUtil.get(locale, monthKey?lower_case) | |
| dateFormat = (locale == 'pt_BR')?then("d 'de ${month} de' yyyy","'${month}' d, yyyy") | |
| formattedDate = previous?date?string(dateFormat) | |
| /] | |
| [#return languageUtil.format(locale, "posted-on-x", formattedDate) /] | |
| [/#function] | |
| [#function getLastUpdateTime previous] | |
| [#assign current = .now | |
| elapsed = current?long - previous?date?long | |
| msPerMinute = 60 * 1000 | |
| msPerHour = msPerMinute * 60 | |
| msPerDay = msPerHour * 24 | |
| msPerMonth = msPerDay * 30 | |
| msPerYear = msPerDay * 365 | |
| /] | |
| [#if elapsed < msPerMinute] | |
| [#assign seconds = (elapsed / 1000)?string('0') /] | |
| [#return "${languageUtil.get(locale, 'last-updated')} ${seconds} ${(seconds == '1')?then(languageUtil.get(locale, 'second'), languageUtil.get(locale, 'seconds'))}" /] | |
| [#elseif elapsed < msPerHour] | |
| [#assign minutes = (elapsed / msPerMinute)?string('0') /] | |
| [#return "${languageUtil.get(locale, 'last-updated')} ${minutes} ${(minutes == '1')?then(languageUtil.get(locale, 'minute'), languageUtil.get(locale, 'minutes'))}" /] | |
| [#elseif elapsed < msPerDay] | |
| [#assign hours = (elapsed / msPerHour)?string('0') /] | |
| [#return "${languageUtil.get(locale, 'last-updated')} ${hours} ${(hours == '1')?then(languageUtil.get(locale, 'hour'), languageUtil.get(locale, 'hours'))}" /] | |
| [#elseif elapsed < msPerMonth] | |
| [#assign days = (elapsed / msPerDay)?string('0') /] | |
| [#return "${languageUtil.get(locale, 'last-updated')} ${days} ${(days == '1')?then(languageUtil.get(locale, 'day'), languageUtil.get(locale, 'days'))}" /] | |
| [#elseif elapsed < msPerYear] | |
| [#assign months = (elapsed / msPerMonth)?string('0') /] | |
| [#return "${languageUtil.get(locale, 'last-updated')} ${months} ${(months == '1')?then(languageUtil.get(locale, 'month'), languageUtil.get(locale, 'months'))}" /] | |
| [#else] | |
| [#assign years = (elapsed / msPerYear)?string('0') /] | |
| [#return "${languageUtil.get(locale, 'last-updated')} ${years} ${(years == '1')?then(languageUtil.get(locale, 'year'), languageUtil.get(locale, 'years'))}" /] | |
| [/#if] | |
| [/#function] |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| if (!configuration.hideShareIcon) { | |
| const shareIcon = fragmentElement.querySelector(".card-share-link"); | |
| shareIcon.addEventListener("click", async () => { | |
| const shareTitle = shareIcon.dataset.shareTitle; | |
| const shareUrl = shareIcon.dataset.shareUrl; | |
| if (navigator.share) { | |
| try { | |
| await navigator.share({ | |
| title: document.title, | |
| text: shareTitle, | |
| url: window.location.host + shareUrl, | |
| }); | |
| } catch (error) { | |
| console.error("Error sharing: ", error); | |
| } | |
| } | |
| }); | |
| } |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment