.ecom-libro {
font-family: Arial, sans-serif;
font-size: 14px;
color: #333;
}
.ecom-libro .main-container,
.ecom-libro .container {
width: 100%;
margin: 40px auto;
display: flex;
gap: 24px;
}
.ecom-libro .main-container {
max-width: 1200px;
}
.ecom-libro .container {
padding: 1em 5em;
}
.ecom-libro .top-section {
display: flex;
align-items: flex-start;
gap: 24px;
}
.ecom-libro .promo-banner {
display: flex;
align-items: center;
justify-content: center;
background: #111;
color: #fff;
font-size: 14px;
padding: 12px 20px;
gap: 8px;
}
.ecom-libro .promo-banner .discount {
color: #ffd700;
font-weight: bold;
}
.ecom-libro .promo-banner .conditions {
display: flex;
align-items: center;
font-weight: bold;
text-decoration: none;
color: #fff;
}
.ecom-libro .promo-banner .conditions:hover {
text-decoration: underline;
}
.ecom-libro .promo-banner .conditions i {
margin-left: 4px;
font-size: 16px;
}
.ecom-libro .product-detail {
flex: 1;
background: #fff;
padding: 24px;
border-radius: 4px;
}
.ecom-libro .header-book {
display: flex;
align-items: flex-start;
gap: 16px;
margin-bottom: 16px;
}
.ecom-libro .header-book .logo-book {
height: auto;
width: auto;
max-height: 400px;
max-width: 300px;
}
.ecom-libro .title {
font-family: 'SohoPro-Regular';
font-size: 24px;
color: #00437a;
margin-top: 0;
font-weight: bold;
}
.ecom-libro .description-book {
margin: 20px 0 10px;
font-size: 1.2em;
font-weight: 600;
line-height: 1.4;
color: #222;
}
.ecom-libro .btn {
display: inline-block;
font-family: 'SohoPro-Regular';
font-weight: bold;
padding: 8px 16px;
text-decoration: none;
}
.ecom-libro .extract-btn {
font-size: 1em !important;
border: 2px solid #00437a;
color: #00437a;
transition: background .2s, color .2s;
text-transform: none !important;
}
.ecom-libro .extract-btn:hover {
background: #e6f0ff;
color: #120e5b;
}
.ecom-libro .purchase-box {
flex: 0 0 300px;
background: #f3f6f9;
padding: 10px 0 15px;
border-radius: 4px;
text-align: center;
}
.ecom-libro .purchase-box .price-header {
padding: 16px;
text-align: left;
}
.ecom-libro .purchase-box .price-header .price {
font-family: 'SohoPro-Regular';
font-size: 28px;
font-weight: bold;
color: #0046ad;
margin-bottom: 0;
}
.ecom-libro .purchase-box .price-header .old-price {
font-family: 'SohoPro-Regular';
font-size: bold;
color: #25323c;
font-size: 20px;
text-decoration: line-through;
margin-bottom: 0;
}
.ecom-libro .purchase-box .price-header .discount {
font-family: 'SohoPro-Regular';
display: inline-block;
background-color: var(--ae-450, #D44436);
color: #fff;
font-size: 1rem;
font-weight: bold;
padding: 0.3em 0.3em;
margin-left: 0.4em;
border-radius: 0.2em;
line-height: 1;
text-align: center;
}
.ecom-libro .purchase-box .price-header .note {
font-family: sans-serif;
font-size: 12px;
font-weight: 500;
color: #000;
margin-bottom: 0;
}
.ecom-libro .purchase-box .selectors {
display: flex;
align-items: center;
gap: 8px;
padding: 0 16px 16px;
}
.ecom-libro .purchase-box .selectors .qty {
flex: 0 0 50px;
}
.ecom-libro .purchase-box .selectors input,
.ecom-libro .purchase-box .selectors select {
width: 100%;
height: 40px;
padding: 0 12px;
font-size: 14px;
border: 1px solid #ccc;
border-radius: 4px;
-webkit-appearance: none;
-moz-appearance: none;
appearance: none;
background: #fff;
}
.ecom-libro .purchase-box .selectors select {
background-image: url(/documents/d/global/ico-chevron-down);
background-repeat: no-repeat;
background-size: 18px 10px;
background-position: right 12px center;
}
.ecom-libro .purchase-box .selectors select::-ms-expand {
display: none;
}
.ecom-libro .purchase-box .selectors .selector {
position: relative;
flex: 1;
}
.ecom-libro .purchase-box .selectors .selector .arrow {
position: absolute;
top: 50%;
right: 12px;
transform: translateY(-50%);
pointer-events: none;
font-size: 12px;
color: #666;
}
.ecom-libro .purchase-box .selectors .selector:focus-within {
background-size: 18px 10px;
background-position-x: 95%;
}
.ecom-libro .purchase-box .buy-btn {
width: calc(100% - 32px);
text-transform: uppercase;
}
.ecom-libro .purchase-box .buy-btn:hover,
.ecom-libro .purchase-box .buy-btn:focus,
.ecom-libro .purchase-box .buy-btn:focus-visible {
outline: 2px solid var(--brand-color-2, #6a9bd3);
outline-offset: 2px;
box-shadow: 0 0 0 3px rgba(106, 155, 211, 0.35);
}
.ecom-libro .purchase-box .buy-btn.disabled,
.ecom-libro .purchase-box .buy-btn:disabled,
.ecom-libro .purchase-box .buy-btn[disabled],
.ecom-libro .purchase-box .buy-btn.ecom-btn-loading {
opacity: .6;
pointer-events: none;
}
.ecom-libro .purchase-box .buy-btn.ecom-btn-loading::before,
.ecom-libro .purchase-box .buy-btn.ecom-btn-loading::after {
display: none;
}
.ecom-libro .tabs-section {
max-width: 900px;
margin: 0 auto 40px;
padding-top: 3em;
background: #fff;
border-radius: 4px;
overflow: hidden;
}
.ecom-libro .tabs-section .tabs {
display: flex;
border-bottom: 1px solid #ddd;
}
.ecom-libro .tabs-section .tab {
padding: 12px 16px;
margin-right: 8px;
background: transparent;
border: none;
border-radius: 4px 4px 0 0;
cursor: pointer;
font-size: 14px;
color: #222;
transition: background .2s;
}
.ecom-libro .tabs-section .tab:hover {
background: rgba(0, 67, 122, 0.05);
}
.ecom-libro .tabs-section .tab.active {
color: #00437a;
background: #e5f5ff;
border-bottom: 2px solid #00437a;
font-weight: bold;
}
.ecom-libro .tabs-section table {
width: 100%;
border-collapse: collapse;
}
.ecom-libro .tabs-section th,
.ecom-libro .tabs-section td {
padding: 12px 16px;
vertical-align: middle;
}
.ecom-libro .tabs-section th {
width: 180px;
font-weight: 600;
color: #444;
text-align: right;
white-space: nowrap;
}
.ecom-libro .tabs-section td {
font-weight: 400;
color: #222;
line-height: 1.4;
}
.ecom-libro .tabs-section tr:nth-child(even) > * {
background: #f9fafb;
}
.ecom-libro .tabs-section .badge {
display: inline-block;
margin-left: 8px;
padding: 2px 6px;
font-size: 11px;
text-transform: uppercase;
border-radius: 3px;
}
.ecom-libro .tabs-section .tag-success {
background: #2a7a36;
color: #fff;
}
.ecom-libro .tabs-section .tag-danger {
background: #c00000;
color: #fff;
}
/* === Media queries === */
@media (max-width: 1024px) {
.ecom-libro .main-container { padding: 20px; margin: 20px auto; }
.ecom-libro .top-section { flex-direction: column; gap: 16px; }
.ecom-libro .purchase-box { flex: none; width: 100%; }
.ecom-libro .product-detail { padding: 16px; }
.ecom-libro .title { font-size: 18px; }
}
@media (max-width: 768px) {
.ecom-libro { font-size: 13px; }
.ecom-libro .promo-banner { flex-direction: column; text-align: center; gap: 4px; padding: 8px 10px; }
.ecom-libro .main-container { margin: 16px auto; }
.ecom-libro .top-section { gap: 12px; }
.ecom-libro .title { font-size: 16px; }
.ecom-libro .extract-btn { padding: 6px 12px; font-size: 13px; }
.ecom-libro .selectors { flex-direction: column; gap: 8px; padding: 12px; }
.ecom-libro .buy-btn { width: 100%; margin-bottom: 16px; }
.ecom-libro .tabs-section .tabs { flex-wrap: wrap; justify-content: center; }
}
.ecom-libro .autores-list {
display: flex;
flex-wrap: wrap;
gap: 8px 12px;
margin-top: 15px;
}
.ecom-libro .autor-item {
display: flex;
align-items: center;
gap: 6px;
background-color: #E6F7FF;
border-radius: 4px;
padding: 4px 8px;
}
.ecom-libro .autor-nombre {
font-size: 1rem;
font-weight: bold;
color: #273469;
}
.ecom-libro .autor-add {
display: inline-flex;
align-items: center;
justify-content: center;
width: 24px;
height: 24px;
font-size: 2em;
color: #4AC8F0;
background: none;
border: none;
cursor: pointer;
border-radius: 4px;
}
.ecom-libro .info-box-book {
display: inline-flex;
width: 100%;
align-items: center;
background-color: #ffffd7;
margin-top: 10px;
padding: 4px 8px;
border-radius: 4px;
color: #273469;
}
.ecom-libro .info-icon {
display: inline-flex;
align-items: center;
justify-content: center;
width: 1.5em;
height: 1.5em;
margin-right: 8px;
border: 2px solid #273469;
border-radius: 50%;
font-size: 1em;
font-weight: bold;
}
.ecom-libro .book-description {
margin-top: 2em;
}
/* ===== estilos de la modal ===== */
.ecom-libro .modal {
position: fixed;
top: 0; left: 0;
width: 100%; height: 100%;
background: rgba(0,0,0,0.5);
display: none;
z-index: 1000;
align-items: center;
justify-content: center;
}
.ecom-libro .modal.show { display: flex; }
.ecom-libro .modal-content {
background: #fff;
border-radius: 8px;
width: 100%;
max-height: 70vh;
overflow-y: auto;
max-width: 600px;
position: relative;
box-shadow: 0 2px 10px rgba(0,0,0,0.2);
font-family: 'SohoPro-Regular';
}
.ecom-libro .modal-header {
font-size: 1.25rem;
font-weight: bold;
border: none;
}
.ecom-libro .modal-close {
position: absolute;
top: 0.5rem;
right: 0.5rem;
background: none;
border: none;
font-size: 1.25rem;
cursor: pointer;
color: #999;
}
.ecom-libro .modal-close:hover {
color: #333;
}
.list-books-section { margin: 40px auto 35px; }
.list-books-section .books-container { display: flex; flex-wrap: wrap; gap: 20px 3.5%; padding-right: 0; padding-left: 0; }
.list-books-section .item-book-adt {
position: relative;
border: 1px solid #e0e0e0;
padding: 28px 14px 16px;
border-radius: 4px;
background: #fff;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
display: flex;
flex-direction: column;
justify-content: space-between;
overflow: visible;
min-height: 1px;
box-sizing: border-box;
}
.list-books-section .item-book-adt .info-book {
position: static;
display: flex;
flex-direction: column;
flex: 1 1 auto;
}
.list-books-section .item-book-adt .info-book a {
display: block;
}
.list-books-section .image-book {
width: 100%;
height: 210px;
object-fit: cover;
}
.list-books-section .title-book {
font-family: SohoGothicPro-Regular;
font-size: 16px;
font-weight: bold;
color: #1a4b94;
margin-top: 10px;
margin-bottom: 6px;
line-height: 24px;
}
.list-books-section .description-text {
margin-top: 0;
margin-bottom: 12px;
font-family: SohoGothicPro-Regular;
font-size: 13px;
text-align: left;
color: #333;
line-height: 20px;
}
.list-books-section .price-container {
display: flex;
flex-direction: column;
text-align: left;
letter-spacing: -0.02em;
margin-top: auto;
}
.list-books-section .price-container .price-ae {
font-size: 20px;
font-weight: bold;
margin-bottom: 10px;
color: #1a4b94;
}
.list-books-section .price-container .old-price {
font-size: 16px;
font-weight: bold;
color: grey;
text-decoration: line-through;
padding-top: 10px;
padding-bottom: 10px;
}
.list-books-section .price-container .old-price-block {
display: inline-flex;
align-items: center;
gap: 8px;
}
.list-books-section .price-container .discount {
background-color: var(--ae-450, #D44436);
color: #fff;
font-size: 16px;
font-weight: bold;
padding: 4px 4px;
border-radius: 2px;
line-height: 1;
}
.list-books-section .book-button {
border: 0;
outline: 0;
padding: 17px;
width: 100%;
text-align: center;
display: block;
background-color: var(--brand-color-1, #1f57a3);
color: #fff;
text-transform: uppercase;
font-family: SohoStd-Medium;
font-size: 14px;
font-weight: 500;
line-height: 14px;
text-decoration: none;
}
.list-books-section .book-button:hover { cursor: pointer; }
.list-books-section .book-button.disabled,
.list-books-section .book-button:disabled,
.list-books-section .book-button[disabled],
.list-books-section .book-button.ecom-btn-loading {
opacity: .6;
pointer-events: none;
filter: brightness(0.85);
}
.list-books-section .books-container .item-book-adt {
width: calc((100% - (3 * 3.5%)) / 4);
}
@media (max-width:1200px){ .list-books-section .books-container .item-book-adt { width: 48%; } }
@media (max-width: 576px) { .list-books-section .books-container .item-book-adt { width: 100%; } }
.list-books-section .options-book {
display: flex;
gap: 10px;
margin-bottom: 16px;
border: none;
}
.list-books-section .options-book .select-language,
.list-books-section .options-book .select-format {
background-color: #f5f5f5;
border: 1px solid #d0d0d0;
color: #2d2d2b;
font-size: 13px;
height: 38px;
}
.list-books-section .options-book .select-language:disabled,
.list-books-section .options-book .select-format:disabled {
background-color: #f5f5f5;
color: #2d2d2b;
opacity: 1;
}
.list-books-section .options-book .select-language { flex: 2; }
.list-books-section .options-book .select-format { flex: 1; }
select.form-control,
select.form-control:focus,
select.form-control:focus-visible {
background-image: url(/documents/d/global/ico-chevron-down);
background-size: 18px 10px;
background-position-x: 95%;
background-repeat: no-repeat;
position: relative;
}
.list-books-section .book-button:hover,
.list-books-section .book-button:focus,
.list-books-section .book-button:focus-visible {
outline: 2px solid var(--brand-color-2, #6a9bd3);
outline-offset: 2px;
box-shadow: 0 0 0 3px rgba(106, 155, 211, 0.35);
}
<script id="ecom-generic-scripts">
window.ecomGenericScripts = window.ecomGenericScripts || (function () {
/* =====================================
PROPIEDADES PRIVADAS
===================================== */
const privateProps = window.ecomGenericScripts?.privateProps || {};
/* =====================================
PROPIEDADES PÚBLICAS
===================================== */
const publicProps = {
isDebug: false
};
/* =====================================
FUNCIONES PÚBLICAS
===================================== */
//En functions declaramos la functions publicas
const functions = {};
functions.init = async function () {
await _DOMContentLoaded();
};
/* =====================================
FUNCIONES PRIVADAS
===================================== */
const _loadInit = async function () {
if (publicProps.isDebug) console.log("DOMContentLoaded ecom Generic scripts");
};
const _parsePriceText = function (text) {
if (!text) return 0;
const cleaned = String(text)
.replace(/[^\d,.-]/g, "")
.replace(/\./g, "")
.replace(",", ".");
const n = Number(cleaned);
return Number.isFinite(n) ? n : 0;
};
const _getDetailContext = function (button) {
const wrapper = document.querySelector("#ecom-scripts .data-product-wrapper");
const purchaseBox = button?.closest(".purchase-box") || document;
const langSelect = purchaseBox.querySelector(".select-language");
const formatSelect = purchaseBox.querySelector(".select-format");
const qtyInput = purchaseBox.querySelector(".qty input[type='number']");
const priceNode = purchaseBox.querySelector(".price-header .price");
const titleNode = document.querySelector(".ecom-libro .title, .ecom-norma .title, .ecom-coleccion_tematica .title, .title-book");
return {
wrapper,
purchaseBox,
langSelect,
formatSelect,
qtyInput,
priceNode,
titleNode
};
};
const _getDetailData = async function (productERC, productId, isLibro, isNorma, isColeccionTematica) {
if (!window.ecomGlobalScripts?.functions) return null;
try {
if (isLibro) {
if (productERC) return await window.ecomGlobalScripts.functions.getProductBooksByERC(productERC);
if (productId) return await window.ecomGlobalScripts.functions.getProductBooks(productId);
}
if (isColeccionTematica) {
if (productERC) return await window.ecomGlobalScripts.functions.getProductThematicCollectionsByERC(productERC);
if (productId) return await window.ecomGlobalScripts.functions.getProductThematicCollections(productId);
}
if (isNorma) {
if (productERC) return await window.ecomGlobalScripts.functions.getProductNormasDetailsByERC(productERC);
if (productId) return await window.ecomGlobalScripts.functions.getProductNormasDetails(productId);
}
} catch (e) {}
return null;
};
const _syncDetailBuyButtonState = function (purchaseBox) {
if (!purchaseBox) return;
const btn = purchaseBox.querySelector(".buy-btn");
if (!btn) return;
const langSelect = purchaseBox.querySelector(".select-language");
const formatSelect = purchaseBox.querySelector(".select-format");
const hasSelectors = !!(langSelect || formatSelect);
const hasSelection = !!(langSelect?.value && formatSelect?.value);
if (!hasSelectors) {
if (!btn.disabled) btn.disabled = true;
if (!btn.classList.contains("disabled")) btn.classList.add("disabled");
return;
}
if (btn.disabled === hasSelection) {
btn.disabled = !hasSelection;
}
btn.classList.toggle("disabled", !hasSelection);
};
const _setupQtyPriceMultiplier = function (purchaseBox) {
if (!purchaseBox || purchaseBox.dataset.aeQtyMultBound === "true") return;
purchaseBox.dataset.aeQtyMultBound = "true";
const priceNode = purchaseBox.querySelector(".price-header .price");
const qtyInput = purchaseBox.querySelector(".qty input[type='number']");
if (!priceNode || !qtyInput) return;
const formatter = window.ecomGlobalScripts?.properties?.language?.formatterPrice;
const splitPriceAndCurrency = function (text) {
const m = String(text || "").match(/^(.*?)(\s*[^\d.,\s]+)\s*$/);
if (m) return [m[1].trim(), m[2].trim()];
return [String(text || "").trim(), ""];
};
const formatPrice = function (value, currency) {
const numText = formatter
? formatter.format(value)
: value.toLocaleString("es-ES", { minimumFractionDigits: 2, maximumFractionDigits: 2 });
return currency ? numText + " " + currency : numText;
};
const captureUnit = function () {
const parts = splitPriceAndCurrency(priceNode.textContent);
const parsed = _parsePriceText(parts[0]);
if (parsed > 0) {
purchaseBox.dataset.aeUnitPrice = String(parsed);
purchaseBox.dataset.aeUnitCurrency = parts[1];
}
};
let observer;
const startObserve = function () {
observer.observe(priceNode, { childList: true, characterData: true, subtree: true });
};
const reapply = function () {
const unit = Number(purchaseBox.dataset.aeUnitPrice) || 0;
if (!unit) return;
const qty = Math.max(1, parseInt(qtyInput.value, 10) || 1);
const total = unit * qty;
const currency = purchaseBox.dataset.aeUnitCurrency || "";
if (observer) observer.disconnect();
priceNode.textContent = formatPrice(total, currency);
if (observer) startObserve();
};
// Cuando _renderFragmentPrice (en ECOM-Global_functions/index.js) reescribe
// .price con el precio unitario tras cambiar idioma/formato, recapturamos
// el unitario y reaplicamos la cantidad actual.
observer = new MutationObserver(function () {
captureUnit();
reapply();
});
startObserve();
qtyInput.addEventListener("input", reapply);
qtyInput.addEventListener("change", reapply);
// Captura inicial por si el precio ya estaba renderizado antes de montar el observer.
captureUnit();
reapply();
};
const _bindDetailSelectorState = function (purchaseBox) {
if (!purchaseBox) return;
const sync = function () {
_syncDetailBuyButtonState(purchaseBox);
};
if (purchaseBox.dataset.aeSelectorsBound === "true") {
sync();
return;
}
purchaseBox.dataset.aeSelectorsBound = "true";
purchaseBox.addEventListener("change", function (event) {
if (!event.target?.matches(".select-language, .select-format")) return;
sync();
});
const selectorsRoot = purchaseBox.querySelector(".selector-language_format") || purchaseBox;
const observer = new MutationObserver(function () {
sync();
});
observer.observe(selectorsRoot, {
childList: true,
subtree: true
});
sync();
};
const _attachDetailAddToCart = async function () {
const buttons = document.querySelectorAll(".purchase-box .buy-btn");
if (!buttons.length) return;
buttons.forEach(btn => {
_bindDetailSelectorState(btn.closest(".purchase-box"));
_setupQtyPriceMultiplier(btn.closest(".purchase-box"));
if (btn.dataset.aeBound === "true") return;
btn.dataset.aeBound = "true";
btn.addEventListener("click", (e) => {
e.preventDefault();
const run = async () => {
const ctx = _getDetailContext(btn);
const wrapper = ctx.wrapper;
if (!wrapper) {
window.dispatchEvent(new CustomEvent("cart:error", { detail: { message: "Error al añadir el producto a la cesta" } }));
return;
}
const productId = Number(wrapper.dataset.productId) || null;
const productERC = wrapper.dataset.productErc || "";
const isLibro = wrapper.dataset.productIslibro === "true";
const isNorma = wrapper.dataset.productIsnorma === "true";
const isColeccionTematica = wrapper.dataset.productIscolecciontematica === "true";
const codIdioma = (ctx.langSelect?.value || "").toString();
const codFormato = (ctx.formatSelect?.value || "").toString();
if ((ctx.langSelect || ctx.formatSelect) && (!codIdioma || !codFormato)) {
window.dispatchEvent(new CustomEvent("cart:error", { detail: { message: "Selecciona idioma y formato" } }));
return;
}
const entryType = isLibro ? "Libro" : isColeccionTematica ? "Colección Temática" : "Norma";
const code = productERC || (ctx.titleNode?.textContent || "").trim();
const qty = Math.max(1, parseInt(ctx.qtyInput?.value, 10) || 1);
const name = (ctx.titleNode?.textContent || "").trim();
let priceToSend = 0;
const data = await _getDetailData(productERC, productId, isLibro, isNorma, isColeccionTematica);
if (Array.isArray(data) && data.length && codIdioma && codFormato) {
const entry = data.find(item => {
const langKey = item.codLanguage ?? item.language;
const fmtKey = item.codFormat ?? item.format;
return String(langKey) === codIdioma && String(fmtKey) === codFormato;
});
if (entry) {
const basePrice = Number(entry.price) || 0;
const discount = Number(entry.webDiscount) || 0;
priceToSend = discount > 0 ? Math.floor((basePrice - (basePrice * discount / 100)) * 100) / 100 : basePrice;
}
}
if (!priceToSend && ctx.priceNode) {
priceToSend = _parsePriceText(ctx.priceNode.textContent);
}
await window.ecomGlobalScripts.functions.addToCart({
entryType: entryType,
code: code,
codIdioma: codIdioma,
codFormato: codFormato,
price: priceToSend,
name: name,
amount: qty
});
};
const helper = window.ecomGlobalScripts?.functions?.withButtonSpinner;
if (typeof helper === "function") {
return helper(btn, run, { spinnerClass: "spinner-border spinner-border-sm text-light", mode: "replace" });
}
return run();
});
});
};
const _DOMContentLoaded = async function() {
//Event DOMContentLoaded
if (publicProps.isDebug) console.log("DOMContentLoaded ecom Generic scripts");
if (publicProps.isDebug) console.log("DOMContentLoaded ecom Generic scripts - _loadInit execute");
await _loadInit();
await _attachDetailAddToCart();
};
// ---- Execute Listener DOMContentLoaded ----
document.addEventListener("DOMContentLoaded", async function() {
await _DOMContentLoaded();
});
/* =====================================
API PÚBLICA
===================================== */
return {
properties: publicProps, // properties public
functions: functions // functions public
};
})();
</script>










