ecommerce/template/catalog/ProductDetail.ftl (1,010 lines of code) (raw):
<#--
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
"License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing,
software distributed under the License is distributed on an
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
KIND, either express or implied. See the License for the
specific language governing permissions and limitations
under the License.
-->
<#-- variable setup -->
<#assign price = priceMap! />
<#assign productImageList = productImageList! />
<#-- end variable setup -->
<#-- virtual product javascript -->
${virtualJavaScript!}
${virtualVariantJavaScript!}
<script type="text/javascript">
var detailImageUrl = null;
function setAddProductId2(sku, with_bak) {
document.addform.add_product_id.value = sku;
var disable = (sku == '' || sku == 'NULL' || isVirtual(sku) == true);
if (document.addform.quantity != null) {
document.addform.quantity.disabled = disable;
}
var txt = disable ? '' : sku;
$('#product_id_display').text(txt);
if (with_bak) {
document.addform.product_id_bak.value = txt;
}
}
function setVariantPrice2(sku) {
var disable = (sku == '' || sku == 'NULL' || isVirtual(sku) == true);
var txt = disable ? '' : getVariantPrice2(sku);
$('#variant_price_display').text(txt || '');
}
function isVirtual(product) {
var isVirtual = false;
<#if virtualJavaScript??>
for (i = 0; i < VIR.length; i++) {
if (VIR[i] == product) {
isVirtual = true;
}
}
</#if>
return isVirtual;
}
function addItem() {
if (document.addform.add_product_id.value == 'NULL') {
showErrorAlert("${uiLabelMap.CommonErrorMessage2}","${uiLabelMap.CommonPleaseSelectAllRequiredOptions}");
return;
} else {
if (isVirtual(document.addform.add_product_id.value)) {
document.location = '<@ofbizUrl>product?category_id=${categoryId!}&product_id=</@ofbizUrl>' + document.addform.add_product_id.value;
return;
} else {
document.addform.submit();
}
}
}
function popupDetail(specificDetailImageUrl) {
if( specificDetailImageUrl ) {
detailImageUrl = specificDetailImageUrl;
}
else {
var defaultDetailImage = "${firstDetailImage?default(mainDetailImageUrl?default("_NONE_"))}";
if (defaultDetailImage == null || "null" == defaultDetailImage || "" == defaultDetailImage) {
defaultDetailImage = "_NONE_";
}
if (detailImageUrl == null || "null" == detailImageUrl) {
detailImageUrl = defaultDetailImage;
}
}
if ("_NONE_" == detailImageUrl) {
hack = document.createElement('span');
hack.innerHTML="${uiLabelMap.CommonNoDetailImageAvailableToDisplay}";
showErrorAlert("${uiLabelMap.CommonErrorMessage2}","${uiLabelMap.CommonNoDetailImageAvailableToDisplay}");
return;
}
detailImageUrl = detailImageUrl.replace(/\&\#47;/g, "/");
popUp("<@ofbizUrl>detailImage?detail=" + detailImageUrl + "</@ofbizUrl>", 'detailImage', '600', '600');
}
function toggleAmt(toggle) {
if (toggle == 'Y') {
changeObjectVisibility("add_amount", "visible");
document.getElementById("add_amount").style.height = "auto";
}
if (toggle == 'N') {
changeObjectVisibility("add_amount", "hidden");
document.getElementById("add_amount").style.height = "0px";
}
}
function findIndex(name) {
for (i = 0; i < OPT.length; i++) {
if (OPT[i] == name) {
return i;
}
}
return -1;
}
function getList(name, index, src) {
currentFeatureIndex = findIndex(name);
if (currentFeatureIndex == 0) {
// set the images for the first selection
if (IMG[index] != null) {
if (document.images['mainImage'] != null) {
document.images['mainImage'].src = IMG[index];
detailImageUrl = DET[index];
}
}
// set the drop down index for swatch selection
document.forms["addform"].elements[name].selectedIndex = (index*1)+1;
}
if (currentFeatureIndex < (OPT.length-1)) {
// eval the next list if there are more
var selectedValue = document.forms["addform"].elements[name].options[(index*1)+1].value;
if (index == -1) {
<#if featureOrderFirst??>
var Variable1 = eval("list" + "${featureOrderFirst}" + "()");
</#if>
} else {
var Variable1 = eval("list" + OPT[(currentFeatureIndex+1)] + selectedValue + "()");
}
// reset uom select
$('#product_uom').text('');
// set the product ID to NULL to trigger the alerts
setAddProductId2('NULL');
// set the variant price to NULL
setVariantPrice2('NULL');
} else {
// this is the final selection -- locate the selected index of the last selection
var indexSelected = document.forms["addform"].elements[name].selectedIndex;
// using the selected index locate the sku
var sku = document.forms["addform"].elements[name].options[indexSelected].value;
// display alternative packaging dropdown
ajaxUpdateArea("product_uom", "<@ofbizUrl>ProductUomDropDownOnly</@ofbizUrl>", "productId=" + sku);
// set the product ID
setAddProductId2(sku, true);
// set the variant price
setVariantPrice2(sku);
// check for amount box
toggleAmt(checkAmtReq2(sku));
}
}
function validate(x){
var msg=new Array();
msg[0]="Please use correct date format [yyyy-mm-dd]";
var y=x.split("-");
if(y.length!=3){ showAlert(msg[0]);return false; }
if((y[2].length>2)||(parseInt(y[2])>31)) { showAlert(msg[0]); return false; }
if(y[2].length==1){ y[2]="0"+y[2]; }
if((y[1].length>2)||(parseInt(y[1])>12)){ showAlert(msg[0]); return false; }
if(y[1].length==1){ y[1]="0"+y[1]; }
if(y[0].length>4){ showAlert(msg[0]); return false; }
if(y[0].length<4) {
if(y[0].length==2) {
y[0]="20"+y[0];
} else {
showAlert(msg[0]);
return false;
}
}
return (y[0]+"-"+y[1]+"-"+y[2]);
}
function showAlert(msg) {
showErrorAlert("${uiLabelMap.CommonErrorMessage2}", msg);
}
function additemSubmit(){
<#if "ASSET_USAGE" == product.productTypeId! || "ASSET_USAGE_OUT_IN" == product.productTypeId!>
newdatevalue = validate(document.addform.reservStart.value);
if (newdatevalue == false) {
document.addform.reservStart.focus();
} else {
document.addform.reservStart.value = newdatevalue;
document.addform.submit();
}
<#else>
document.addform.submit();
</#if>
}
function addShoplistSubmit(){
<#if "ASSET_USAGE" == product.productTypeId! || "ASSET_USAGE_OUT_IN" == product.productTypeId!>
if ("" == document.addToShoppingList.reservStartStr.value) {
document.addToShoppingList.submit();
} else {
newdatevalue = validate(document.addToShoppingList.reservStartStr.value);
if (newdatevalue == false) {
document.addToShoppingList.reservStartStr.focus();
} else {
document.addToShoppingList.reservStartStr.value = newdatevalue;
// document.addToShoppingList.reservStart.value = ;
document.addToShoppingList.reservStartStr.value.slice(0,9)+" 00:00:00.000000000";
document.addToShoppingList.submit();
}
}
<#else>
document.addToShoppingList.submit();
</#if>
}
<#if "VV_FEATURETREE" == product.virtualVariantMethodEnum! && featureLists?has_content>
function checkRadioButton() {
var block1 = document.getElementById("addCart1");
var block2 = document.getElementById("addCart2");
<#list featureLists as featureList>
<#list featureList as feature>
<#if feature_index == 0>
var myList = document.getElementById("FT${feature.productFeatureTypeId}");
if (myList.options[0].selected == true){
block1.style.display = "none";
block2.style.display = "block";
return;
}
<#break>
</#if>
</#list>
</#list>
block1.style.display = "block";
block2.style.display = "none";
}
</#if>
$(function(){
$('a[id^=productTag_]').click(function(){
var id = $(this).attr('id');
var ids = id.split('_');
var productTagStr = ids[1];
if (productTagStr) {
$('#productTagStr').val(productTagStr);
$('#productTagsearchform').submit();
}
});
})
</script>
<#macro showUnavailableVarients>
<#if unavailableVariants??>
<ul>
<#list unavailableVariants as prod>
<#assign features = prod.getRelated("ProductFeatureAppl", null, null, false)/>
<li>
<#list features as feature>
<em>${feature.getRelatedOne("ProductFeature", false).description}</em><#if feature_has_next>, </#if>
</#list>
<span>${uiLabelMap.ProductItemOutOfStock}</span>
</li>
</#list>
</ul>
</#if>
</#macro>
${screens.render("component://order/widget/ordermgr/OrderEntryCatalogScreens.xml#productvariantjs")}
${variantInfoJavaScript!}
<div id="product-detail" class="card">
<#assign productAdditionalImage1 = productContentWrapper.get("XTRA_IMG_1_MEDIUM", "url")! />
<#assign productAdditionalImage2 = productContentWrapper.get("XTRA_IMG_2_MEDIUM", "url")! />
<#assign productAdditionalImage3 = productContentWrapper.get("XTRA_IMG_3_MEDIUM", "url")! />
<#assign productAdditionalImage4 = productContentWrapper.get("XTRA_IMG_4_MEDIUM", "url")! />
<#-- Category next/previous -->
<#if category??>
<div class="row">
<div class="col-auto mt-3 ml-2">
<#if previousProductId??>
<a class="btn btn-outline-secondary" href="<@ofbizCatalogAltUrl productCategoryId=categoryId! productId=previousProductId!/>">${uiLabelMap.CommonPrevious}</a>
</#if>
</div>
<#if category.categoryName?has_content>
<div class="col-auto"
<a href="<@ofbizCatalogAltUrl productCategoryId=categoryId!/>" class="linktext">${(category.categoryName)?default(category.description)!}</a>
</div>
</#if>
<div class="col-auto mt-3">
<#if nextProductId??>
<a class="btn btn-outline-secondary" href="<@ofbizCatalogAltUrl productCategoryId=categoryId! productId=nextProductId!/>">${uiLabelMap.CommonNext}</a>
</#if>
</div>
</div>
</#if>
<hr/>
<div class="card-body">
<div class="row">
<div class="col-lg-4">
<#if productImageList?has_content>
<#-- Product image/name/price -->
<div class="detail-image">
<#assign productLargeImageUrl = productContentWrapper.get("LARGE_IMAGE_URL", "url")! />
<#-- remove the next two lines to always display the virtual image first (virtual images must exist) -->
<#if firstLargeImage?has_content>
<#assign productLargeImageUrl = firstLargeImage />
</#if>
<#if productLargeImageUrl?string?has_content>
<a href="javascript:popupDetail();">
<img id="detailImage" src="<@ofbizContentUrl>${contentPathPrefix!}${productLargeImageUrl!}</@ofbizContentUrl>"
name="mainImage" vspace="5" hspace="5" class="cssImgXLarge" alt=""/>
</a>
<input type="hidden" id="originalImage" name="originalImage"
value="<@ofbizContentUrl>${contentPathPrefix!}${productLargeImageUrl!}</@ofbizContentUrl>"/>
</#if>
<#if !productLargeImageUrl?string?has_content>
<img id="detailImage" src="/images/defaultImage.jpg" name="mainImage" alt=""/>
</#if>
</div>
<#-- Show Image Approved -->
<#if productImageList?has_content>
<ul class="list-inline gallery">
<#list productImageList as productImage>
<li class="list-inline-item">
<a href="javascript:void(0);"
swapDetail="<@ofbizContentUrl>${productImage.productImage}</@ofbizContentUrl>">
<img src="<@ofbizContentUrl>${productImage.productImageThumb}</@ofbizContentUrl>"
vspace="5" hspace="5" alt=""/>
</a>
</li>
</#list>
</ul>
</#if>
<#else>
<#-- Product image/name/price -->
<div id="detailImageBox">
<#assign productLargeImageUrl = productContentWrapper.get("LARGE_IMAGE_URL", "url")! />
<#-- remove the next two lines to always display the virtual image first (virtual images must exist) -->
<#if firstLargeImage?has_content>
<#assign productLargeImageUrl = firstLargeImage />
</#if>
<#if productLargeImageUrl?string?has_content>
<a href="javascript:popupDetail();">
<img id="detailImage" src="<@ofbizContentUrl>${contentPathPrefix!}${productLargeImageUrl!}</@ofbizContentUrl>"
name="mainImage" vspace="5" hspace="5" class="cssImgXLarge" alt=""/>
</a>
<input type="hidden" id="originalImage" name="originalImage"
value="<@ofbizContentUrl>${contentPathPrefix!}${productLargeImageUrl!}</@ofbizContentUrl>"/>
</#if>
<#if !productLargeImageUrl?string?has_content>
<img id="detailImage" src="/images/defaultImage.jpg" name="mainImage" alt=""/>
</#if>
</div>
<div id="additionalImageBox">
<#if productAdditionalImage1?string?has_content>
<div class="additionalImage">
<a href="javascript:void(0);"
swapDetail="<@ofbizContentUrl>${productAdditionalImage1}</@ofbizContentUrl>">
<img src="<@ofbizContentUrl>${productAdditionalImage1}</@ofbizContentUrl>" vspace="5" hspace="5"
class="cssImgXLarge" alt=""/>
</a>
</div>
</#if>
<#if productAdditionalImage2?string?has_content>
<div class="additionalImage">
<a href="javascript:void(0);"
swapDetail="<@ofbizContentUrl>${productAdditionalImage2}</@ofbizContentUrl>">
<img src="<@ofbizContentUrl>${productAdditionalImage2}</@ofbizContentUrl>" vspace="5" hspace="5"
class="cssImgXLarge" alt=""/>
</a>
</div>
</#if>
<#if productAdditionalImage3?string?has_content>
<div class="additionalImage">
<a href="javascript:void(0);"
swapDetail="<@ofbizContentUrl>${productAdditionalImage3}</@ofbizContentUrl>">
<img src="<@ofbizContentUrl>${productAdditionalImage3}</@ofbizContentUrl>" vspace="5" hspace="5"
class="cssImgXLarge" alt=""/>
</a>
</div>
</#if>
<#if productAdditionalImage4?string?has_content>
<div class="additionalImage">
<a href="javascript:void(0);"
swapDetail="<@ofbizContentUrl>${productAdditionalImage4}</@ofbizContentUrl>">
<img src="<@ofbizContentUrl>${productAdditionalImage4}</@ofbizContentUrl>" vspace="5" hspace="5"
class="cssImgXLarge" alt=""/>
</a>
</div>
</#if>
</div>
</#if>
</div>
<div class="col-lg-8">
<h2>${productContentWrapper.get("PRODUCT_NAME", "html")!}</h2>
<p>${productContentWrapper.get("DESCRIPTION", "html")!}</p>
<p>${product.productId!}</p>
<#-- example of showing a certain type of feature with the product -->
<#if sizeProductFeatureAndAppls?has_content>
<div>
<#if (sizeProductFeatureAndAppls?size == 1)>
${uiLabelMap.OrderSizeAvailableSingle}:
<#else>
${uiLabelMap.OrderSizeAvailableMultiple}:
</#if>
<#list sizeProductFeatureAndAppls as sizeProductFeatureAndAppl>
${sizeProductFeatureAndAppl.description?default(
sizeProductFeatureAndAppl.abbrev?default(sizeProductFeatureAndAppl.productFeatureId))}
<#if sizeProductFeatureAndAppl_has_next>,</#if>
</#list>
</div>
</#if>
<#-- for prices:
- if price < competitivePrice, show competitive or "Compare At" price
- if price < listPrice, show list price
- if price < defaultPrice and defaultPrice < listPrice, show default
- if isSale show price with salePrice style and print "On Sale!"
-->
<#if price.competitivePrice?? && price.price?? && price.price < price.competitivePrice>
<div>${uiLabelMap.ProductCompareAtPrice}:
<span class="basePrice">
<@ofbizCurrency amount=price.competitivePrice isoCode=price.currencyUsed />
</span>
</div>
</#if>
<#if price.listPrice?? && price.price?? && price.price < price.listPrice>
<div>${uiLabelMap.ProductListPrice}:
<span class="basePrice">
<@ofbizCurrency amount=price.listPrice isoCode=price.currencyUsed />
</span>
</div>
</#if>
<#if price.listPrice?? && price.defaultPrice?? && price.price?? &&
price.price < price.defaultPrice && price.defaultPrice < price.listPrice>
<div>
${uiLabelMap.ProductRegularPrice}:
<span class="basePrice">
<@ofbizCurrency amount=price.defaultPrice isoCode=price.currencyUsed />
</span>
</div>
</#if>
<#if price.specialPromoPrice??>
<div>${uiLabelMap.ProductSpecialPromoPrice}:
<span class="basePrice">
<@ofbizCurrency amount=price.specialPromoPrice isoCode=price.currencyUsed />
</span>
</div>
</#if>
<div>
<strong>
<#if price.isSale?? && price.isSale>
<span class="salePrice">${uiLabelMap.OrderOnSale}!</span>
<#assign priceStyle = "salePrice" />
<#else>
<#assign priceStyle = "regularPrice" />
</#if>
${uiLabelMap.OrderYourPrice}:
<#if "Y" = product.isVirtual!>
${uiLabelMap.CommonFrom}
</#if>
<span class="${priceStyle}">
<@ofbizCurrency amount=price.price isoCode=price.currencyUsed />
</span>
<#if "ASSET_USAGE" == product.productTypeId! || "ASSET_USAGE_OUT_IN" == product.productTypeId!>
<#if product.reserv2ndPPPerc?? && product.reserv2ndPPPerc != 0><br/>
<span class="${priceStyle}">
${uiLabelMap.ProductReserv2ndPPPerc}
<#if !product.reservNthPPPerc?? || product.reservNthPPPerc == 0>
${uiLabelMap.CommonUntil} ${product.reservMaxPersons!1}
</#if>
<@ofbizCurrency amount=product.reserv2ndPPPerc*price.price/100 isoCode=price.currencyUsed />
</span>
</#if>
<#if product.reservNthPPPerc?? &&product.reservNthPPPerc != 0><br/>
<span class="${priceStyle}">
${uiLabelMap.ProductReservNthPPPerc}
<#if !product.reserv2ndPPPerc?? || product.reserv2ndPPPerc == 0>
${uiLabelMap.ProductReservSecond}
<#else>
${uiLabelMap.ProductReservThird}
</#if>
${uiLabelMap.CommonUntil} ${product.reservMaxPersons!1}, ${uiLabelMap.ProductEach}:
<@ofbizCurrency amount=product.reservNthPPPerc*price.price/100 isoCode=price.currencyUsed />
</span>
</#if>
<#if (!product.reserv2ndPPPerc?? || product.reserv2ndPPPerc == 0) && (!product.reservNthPPPerc?? ||
product.reservNthPPPerc == 0)>
<br/>${uiLabelMap.ProductMaximum} ${product.reservMaxPersons!1} ${uiLabelMap.ProductPersons}.
</#if>
</#if>
</strong>
</div>
<#if price.listPrice?? && price.price?? && price.price < price.listPrice>
<#assign priceSaved = price.listPrice - price.price />
<#assign percentSaved = (priceSaved / price.listPrice) * 100 />
<div>
${uiLabelMap.OrderSave}:
<span class="basePrice">
<@ofbizCurrency amount=priceSaved isoCode=price.currencyUsed />
(${percentSaved?int}%)
</span>
</div>
</#if>
<#-- show price details ("showPriceDetails" field can be set in the screen definition) -->
<#if (showPriceDetails?? && "Y" == showPriceDetails?default("N"))>
<#if price.orderItemPriceInfos??>
<#list price.orderItemPriceInfos as orderItemPriceInfo>
<div>${orderItemPriceInfo.description!}</div>
</#list>
</#if>
</#if>
<#-- Included quantities/pieces -->
<#if product.piecesIncluded?? && product.piecesIncluded?long != 0>
<div>
${uiLabelMap.OrderPieces}: ${product.piecesIncluded}
</div>
</#if>
<#if (product.quantityIncluded?? && product.quantityIncluded != 0) || product.quantityUomId?has_content>
<#assign quantityUom = product.getRelatedOne("QuantityUom", true)! />
<div>
${uiLabelMap.CommonQuantity} :
${product.quantityIncluded!} ${((quantityUom.abbreviation)?default(product.quantityUomId))!}
</div>
</#if>
<#if (product.productWeight?? && product.productWeight != 0) || product.weightUomId?has_content>
<#assign weightUom = product.getRelatedOne("WeightUom", true)! />
<div>
${uiLabelMap.CommonWeight}:
${product.productWeight!} ${((weightUom.abbreviation)?default(product.weightUomId))!}
</div>
</#if>
<#if (product.productHeight?? && product.productHeight != 0) || product.heightUomId?has_content>
<#assign heightUom = product.getRelatedOne("HeightUom", true)! />
<div>
${uiLabelMap.CommonHeight}:
${product.productHeight!} ${((heightUom.abbreviation)?default(product.heightUomId))!}
</div>
</#if>
<#if (product.productWidth?? && product.productWidth != 0) || product.widthUomId?has_content>
<#assign widthUom = product.getRelatedOne("WidthUom", true)! />
<div>
${uiLabelMap.CommonWidth}:
${product.productWidth!} ${((widthUom.abbreviation)?default(product.widthUomId))!}
</div>
</#if>
<#if (product.productDepth?? && product.productDepth != 0) || product.depthUomId?has_content>
<#assign depthUom = product.getRelatedOne("DepthUom", true)! />
<div>
${uiLabelMap.CommonDepth}:
${product.productDepth!} ${((depthUom.abbreviation)?default(product.depthUomId))!}
</div>
</#if>
<#if daysToShip??>
<div>
<strong>
${uiLabelMap.ProductUsuallyShipsIn} ${daysToShip} ${uiLabelMap.CommonDays}!
</strong>
</div>
</#if>
<#-- show tell a friend details only in ecommerce application -->
<div> </div>
<div>
<a href="javascript:popUpSmall('<@ofbizUrl>tellafriend?productId=${product.productId}<#if categoryId??>&categoryId=${categoryId}/</#if></@ofbizUrl>','tellafriend');"
>${uiLabelMap.CommonTellAFriend}</a>
</div>
<#if disFeatureList?? && 0 < disFeatureList.size()>
<p> </p>
<#list disFeatureList as currentFeature>
<#assign disFeatureType = currentFeature.getRelatedOne("ProductFeatureType", true) />
<div>
<#if disFeatureType.description??>
${disFeatureType.get("description", locale)}
<#else>
${currentFeature.productFeatureTypeId}
</#if>:
${currentFeature.description}
</div>
</#list>
<div> </div>
</#if>
<div id="addItemForm">
<form method="post" action="<@ofbizUrl>additem</@ofbizUrl>" name="addform" style="margin: 0;">
<#assign inStock = true />
<#assign commentEnable = Static["org.apache.ofbiz.entity.util.EntityUtilProperties"]
.getPropertyValue("order", "order.item.comment.enable", delegator)>
<#if commentEnable.equals("Y")>
<#assign orderItemAttr = Static["org.apache.ofbiz.entity.util.EntityUtilProperties"]
.getPropertyValue("order", "order.item.attr.prefix", delegator)>
<div class="form-group">
<label for="${orderItemAttr}comment"> ${uiLabelMap.CommonComment} </label> <input type="text" class="form-control" name="${orderItemAttr}comment" id="${orderItemAttr}comment"/>
</div>
</#if>
<#-- Variant Selection -->
<#if "Y" == product.isVirtual!?upper_case>
<#if "VV_FEATURETREE" == product.virtualVariantMethodEnum! && featureLists?has_content>
<#list featureLists as featureList>
<div class="form-group">
<#list featureList as feature>
<#if feature_index == 0>
${feature.description}:
<select id="FT${feature.productFeatureTypeId}" name="FT${feature.productFeatureTypeId}" class="form-control"
onchange="javascript:checkRadioButton();">
<option value="select" selected="selected">
${uiLabelMap.EcommerceSelectOption}
</option>
<#else>
<option value="${feature.productFeatureId}">
${feature.description}
<#if feature.price??>
(+ <@ofbizCurrency amount=feature.price?string isoCode=feature.currencyUomId />)
</#if>
</option>
</#if>
</#list>
</select>
</div>
</#list>
<input type="hidden" name="add_product_id" value="${product.productId}"/>
<div id="addCart1" class="form-group" style="display:none;">
<label><strong>${uiLabelMap.CommonQuantity}:</strong></label>
<input type="text" class="form-control" size="5" name="quantity" value="1"/>
<a href="javascript:javascript:addItem();" class="btn btn-outline-secondary"><span
style="white-space: nowrap;">${uiLabelMap.OrderAddToCart}</span></a>
</div>
<div id="addCart2" style="display:block;">
<span style="white-space: nowrap;"><strong>${uiLabelMap.CommonQuantity}:</strong></span>
<input type="text" class="form-control" size="5" value="1" disabled="disabled"/>
<a href="javascript:showErrorAlert('${uiLabelMap.CommonErrorMessage2}','${uiLabelMap.CommonPleaseSelectAllFeaturesFirst}');"
class="btn btn-outline-secondary"><span style="white-space: nowrap;">${uiLabelMap.OrderAddToCart}</span></a>
</div>
</#if>
<#if !product.virtualVariantMethodEnum?? || "VV_VARIANTTREE" == product.virtualVariantMethodEnum>
<#if variantTree?? && (variantTree.size() > 0)>
<#list featureSet as currentType>
<div class="form-group">
<select name="FT${currentType}" class="form-control" onchange="javascript:getList(this.name, (this.selectedIndex-1), 1);">
<option>${featureTypes.get(currentType)}</option>
</select>
</div>
</#list>
<span id="product_uom"></span><br/>
<input type="hidden" name="product_id_bak" value=""/>
<div class="variant-price" style="display: inline-block;">
<strong><span id="product_id_display" class="product_id_display"> </span></strong>
<strong><span id="variant_price_display" class="variant_price_display"> </span></strong>
</div>
<input type="hidden" name="product_id" value="${product.productId}"/>
<input type="hidden" name="add_product_id" value="NULL"/>
<#else>
<input type="hidden" name="add_product_id" value="NULL"/>
<#assign inStock = false />
</#if>
</#if>
<#else>
<input type="hidden" name="add_product_id" value="${product.productId}"/>
<#if mainProducts?has_content>
<input type="hidden" name="product_id" value=""/>
<select name="productVariantId" class="form-control" onchange="javascript:variantUomSelection(this);">
<option value="">${uiLabelMap.CommonSelect} ${uiLabelMap.ProductUnitOfMeasure}</option>
<#list mainProducts as mainProduct>
<option value="${mainProduct.productId}">${mainProduct.uomDesc} : ${mainProduct.piecesIncluded}</option>
</#list>
</select><br/>
<input type="hidden" name="product_id_bak" value="${product.productId}"/>
<div class="variant-price" style="display: inline-block;">
<strong><span class="product_id_display"> </span></strong>
<strong><span class="variant_price_display"> </span></strong>
</div>
</#if>
<#if (availableInventory??) && (availableInventory <= 0) && "N" == product.requireAmount?default("N")>
<#assign inStock = false />
</#if>
</#if>
</div>
<#-- check to see if introductionDate hasnt passed yet -->
<#if product.introductionDate?? && nowTimestamp.before(product.introductionDate)>
<div style="color: red;">${uiLabelMap.ProductProductNotYetMadeAvailable}.</div>
<#-- check to see if salesDiscontinuationDate has passed -->
<#elseif product.salesDiscontinuationDate?? && nowTimestamp.after(product.salesDiscontinuationDate)>
<div style="color: red;">${uiLabelMap.ProductProductNoLongerAvailable}.</div>
<#-- check to see if the product requires inventory check and has inventory -->
<#elseif product.virtualVariantMethodEnum! != "VV_FEATURETREE">
<#if inStock>
<#if "Y" == product.requireAmount?default("N")>
<#assign hiddenStyle = "visible" />
<#else>
<#assign hiddenStyle = "hidden"/>
</#if>
<div id="add_amount" class="${hiddenStyle} form-group">
<label>${uiLabelMap.CommonAmount}:</label>
<input type="text" class="form-control" name="add_amount" value=""/>
</div>
<#if "ASSET_USAGE" == product.productTypeId! || "ASSET_USAGE_OUT_IN" == product.productTypeId!>
<div>
<label>
Start Date(yyyy-mm-dd)
</label>
<@htmlTemplate.renderDateTimeField event="" action="" name="reservStart" className="" alert=""
title="Format: yyyy-MM-dd HH:mm:ss.SSS" value="${startDate}" size="25" maxlength="30"
id="reservStart1" isDateType=true shortDateInput=true timeDropdownParamName=""
defaultDateTimeString="" localizedIconTitle="" timeDropdown="" timeHourName="" classString=""
hour1="" hour2="" timeMinutesName="" minutes="" isTwelveHour="" ampmName="" amSelected=""
pmSelected="" compositeType="" formName=""/>
</div>
<div>
<#--td nowrap="nowrap" align="right">Number<br />of days</td>
<td><input type="text" size="4" name="reservLength"/></td></tr>
<tr><td> </td><td align="right" nowrap="nowrap"> </td-->
Number of days<input type="text" class="form-control" size="4" name="reservLength" value=""/>
Number of persons<input type="text" class="form-control" size="4" name="reservPersons" value="2"/>
Number of rooms<input type="text" class="form-control" size="5" name="quantity" value="1"/>
</div>
<a href="javascript:addItem()" class="btn btn-outline-secondary"><span
style="white-space: nowrap;">${uiLabelMap.OrderAddToCart}</span></a>
<#else>
<div class="form-group">
<label>${uiLabelMap.CommonQuantity}:</label>
<div class="input-group">
<input name="quantity" class="form-control" id="quantity" value="1" size="4" maxLength="4" type="text"
<#if "Y" == product.isVirtual!?upper_case>disabled="disabled"</#if>/><span class="input-group-btn">
<a href="javascript:addItem()" id="addToCart" name="addToCart" class="btn btn-outline-secondary">${uiLabelMap.OrderAddToCart}</a></span>
</div>
</div>
<@showUnavailableVarients/>
</#if>
<#else>
<#if productStore??>
<#if productStore.requireInventory?? && "N" == productStore.requireInventory>
<div class="input-group"><input name="quantity" class="form-control" id="quantity" value="1" size="4" maxLength="4" type="text"
<#if "Y" == product.isVirtual!?upper_case>disabled="disabled"</#if>/><a
href="javascript:addItem()" id="addToCart" name="addToCart"
class="btn btn-outline-secondary">${uiLabelMap.OrderAddToCart}</a></div>
<@showUnavailableVarients/>
<#else>
<div class="input-group"><input name="quantity" class="form-control" id="quantity" value="1" size="4" maxLength="4" type="text"
disabled="disabled"/><a href="javascript:void(0);" disabled="disabled"
class="btn btn-outline-secondary">${uiLabelMap.OrderAddToCart}</a></div><br/>
<span>${uiLabelMap.ProductItemOutOfStock}<#if product.inventoryMessage??>— ${product.inventoryMessage}</#if></span>
</#if>
</#if>
</#if>
</#if>
<#if variantPriceList??>
<#list variantPriceList as vpricing>
<#assign variantName = vpricing.get("variantName")!>
<#assign secondVariantName = vpricing.get("secondVariantName")!>
<#assign minimumQuantity = vpricing.get("minimumQuantity")>
<#if minimumQuantity > 0>
<div>minimum order quantity for ${secondVariantName!} ${variantName!} is ${minimumQuantity!}</div>
</#if>
</#list>
<#elseif minimumQuantity?? && minimumQuantity?has_content && minimumQuantity > 0>
<div>minimum order quantity for ${productContentWrapper.get("PRODUCT_NAME", "html")!}
is ${minimumQuantity!}</div>
</#if>
</form>
</div>
<div>
<#if sessionAttributes.userLogin?has_content && sessionAttributes.userLogin.userLoginId != "anonymous">
<form name="addToShoppingList" method="post"
action="<@ofbizUrl>addItemToShoppingList<#if requestAttributes._CURRENT_VIEW_??>/${requestAttributes._CURRENT_VIEW_}</#if></@ofbizUrl>">
<fieldset>
<input type="hidden" name="productId" value="${product.productId}"/>
<input type="hidden" name="product_id" value="${product.productId}"/>
<input type="hidden" name="productStoreId" value="${productStoreId}"/>
<input type="hidden" name="reservStart" value=""/>
<select name="shoppingListId" class="form-control">
<#if shoppingLists?has_content>
<#list shoppingLists as shoppingList>
<option value="${shoppingList.shoppingListId}">${shoppingList.listName}</option>
</#list>
</#if>
<option value="">---</option>
<option value="">${uiLabelMap.OrderNewShoppingList}</option>
</select>
<#--assign nowDate = Static["org.apache.ofbiz.base.util.UtilDateTime"].nowDateString("yyyy-MM-dd")-->
<#if "ASSET_USAGE" == product.productTypeId!>
${uiLabelMap.CommonStartDate}(yyyy-mm-dd)
<@htmlTemplate.renderDateTimeField name="reservStartStr" event="" action="" value="${startDate}"
className="" alert="" title="Format: yyyy-MM-dd HH:mm:ss.SSS" size="15" maxlength="30"
id="reservStartStr" isDateType=true shortDateInput=false timeDropdownParamName=""
defaultDateTimeString="" localizedIconTitle="" timeDropdown="" timeHourName="" classString="" hour1=""
hour2="" timeMinutesName="" minutes="" isTwelveHour="" ampmName="" amSelected="" pmSelected=""
compositeType="" formName=""/> Number of days
<input type="text" size="4" class="form-control" name="reservLength"/> <br/>Number of persons
<input type="text" size="4" class="form-control" name="reservPersons" value="1"/> Qty
<div class="input-group">
<input type="text" size="5" class="form-control" name="quantity" value="1"/>
<#elseif "ASSET_USAGE_OUT_IN" == product.productTypeId!>
${uiLabelMap.CommonStartDate}(yyyy-mm-dd)
<@htmlTemplate.renderDateTimeField name="reservStartStr" event="" action="" value="${startDate}"
className="" alert="" title="Format: yyyy-MM-dd HH:mm:ss.SSS" size="15" maxlength="30"
id="reservStartStr" isDateType=true shortDateInput=false timeDropdownParamName=""
defaultDateTimeString="" localizedIconTitle="" timeDropdown="" timeHourName="" classString="" hour1=""
hour2="" timeMinutesName="" minutes="" isTwelveHour="" ampmName="" amSelected="" pmSelected=""
compositeType="" formName=""/> Number of days
<input type="text" class="form-control" size="4" name="reservLength"/>
<input type="hidden" size="4" name="reservPersons" value="1"/><br/>Qty
<div class="input-group">
<input type="text" class="form-control" size="5" name="quantity" value="1"/>
<#else>
<div class="input-group">
<input type="text" class="form-control" size="5" name="quantity" value="1"/>
<input type="hidden" name="reservStartStr" value=""/>
</#if>
<a href="javascript:addShoplistSubmit();" class="btn btn-outline-secondary">${uiLabelMap.OrderAddToShoppingList}</a>
</div>
</fieldset>
</form>
<#else> <br/>
${uiLabelMap.OrderYouMust}
<a href="<@ofbizUrl>checkLogin/showcart</@ofbizUrl>">${uiLabelMap.CommonBeLogged}</a>
${uiLabelMap.OrderToAddSelectedItemsToShoppingList}.
</#if>
</div>
<#-- Prefill first select box (virtual products only) -->
<#if variantTree?? && 0 < variantTree.size()>
<script type="text/javascript">eval("list" + "${featureOrderFirst}" + "()");</script>
</#if>
<#-- Swatches (virtual products only) -->
<#if variantSample?? && 0 < variantSample.size()>
<#assign imageKeys = variantSample.keySet() />
<#assign imageMap = variantSample />
<p> </p>
<#assign maxIndex = 7 />
<#assign indexer = 0 />
<ul class="list-inline">
<#list imageKeys as key>
<#assign swatchProduct = imageMap.get(key) />
<#if swatchProduct?has_content && indexer < maxIndex>
<#assign imageUrl = Static["org.apache.ofbiz.product.product.ProductContentWrapper"].getProductContentAsText(swatchProduct, "SMALL_IMAGE_URL", request, "url")! />
<#if !imageUrl?string?has_content>
<#assign imageUrl = productContentWrapper.get("SMALL_IMAGE_URL", "url")! />
</#if>
<#if !imageUrl?string?has_content>
<#assign imageUrl = "/images/defaultImage.jpg" />
</#if>
<li class="list-inline-item">
<a href="javascript:getList('FT${featureOrderFirst}','${indexer}',1);" class="linktext">${key}</a>
<a href="javascript:getList('FT${featureOrderFirst}','${indexer}',1);">
<img src="<@ofbizContentUrl>${contentPathPrefix!}${imageUrl}</@ofbizContentUrl>" class="cssImgSmall" alt=""/>
</a>
</li>
</#if>
<#assign indexer = indexer + 1 />
</#list>
</ul>
<#if (indexer > maxIndex)>
<div><strong>${uiLabelMap.ProductMoreOptions}</strong></div>
</#if>
</#if>
<#-- Digital Download Files Associated with this Product -->
<#if downloadProductContentAndInfoList?has_content>
<div id="download-files">
<div>${uiLabelMap.OrderDownloadFilesTitle}:</div>
<#list downloadProductContentAndInfoList as downloadProductContentAndInfo>
<div>${downloadProductContentAndInfo.contentName!}<#if downloadProductContentAndInfo.description?has_content>
- ${downloadProductContentAndInfo.description}</#if></div>
</#list>
</div>
</#if>
<#-- Long description of product -->
<div id="long-description">
<div>${productContentWrapper.get("LONG_DESCRIPTION", "html")!}</div>
<div>${productContentWrapper.get("WARNINGS", "html")!}</div>
</div>
</div>
</div>
<#-- Any attributes/etc may go here -->
<#-- Product Reviews -->
<hr>
<div id="reviews">
<h4>${uiLabelMap.OrderCustomerReviews}:</h4>
<#if averageRating?? && (averageRating > 0) && numRatings?? && (numRatings > 1)>
<div>${uiLabelMap.OrderAverageRating}: ${averageRating} <#if numRatings??>
(${uiLabelMap.CommonFrom} ${numRatings} ${uiLabelMap.OrderRatings})</#if></div>
</#if>
<#if productReviews?has_content>
<#list productReviews as productReview>
<#assign postedUserLogin = productReview.getRelatedOne("UserLogin", false) />
<#assign postedPerson = postedUserLogin.getRelatedOne("Person", false)! />
<div>
<strong>${uiLabelMap.CommonBy} : </strong>
<#if "Y" == productReview.postedAnonymous?default("N")>
${uiLabelMap.OrderAnonymous}
<#else>
${postedPerson.firstName} ${postedPerson.lastName}
</#if>
</div>
<div><strong>${uiLabelMap.CommonAt}: </strong>${productReview.postedDateTime!} </div>
<div><strong>${uiLabelMap.OrderRanking}: </strong>${productReview.productRating!?string}</div>
<div> </div>
<div>${productReview.productReview!}</div>
<hr/>
</#list>
<div>
<a href="<@ofbizUrl>reviewProduct?category_id=${categoryId!}&product_id=${product.productId}</@ofbizUrl>"
class="linktext">${uiLabelMap.ProductReviewThisProduct}!</a>
</div>
<#else>
<p>${uiLabelMap.ProductProductNotReviewedYet}. <a href="<@ofbizUrl>reviewProduct?category_id=${categoryId!}&product_id=${product.productId}</@ofbizUrl>" class="linktext">${uiLabelMap.ProductBeTheFirstToReviewThisProduct}</a>
</p>
</#if>
</div>
<#-- Upgrades/Up-Sell/Cross-Sell -->
<#macro associated assocProducts beforeName showName afterName formNamePrefix targetRequestName>
<#local pageProduct = product />
<#local targetRequest = "product" />
<#if targetRequestName?has_content>
<#local targetRequest = targetRequestName />
</#if>
<#if assocProducts?has_content>
<h2>
${beforeName!}
<#if "Y" == showName>${productContentWrapper.get("PRODUCT_NAME", "html")!}</#if>${afterName!}
</h2>
<div class="productsummary-container">
<div class="row">
<#list assocProducts as productAssoc>
<#if productAssoc.productId == product.productId>
<#local assocProductId = productAssoc.productIdTo />
<#else>
<#local assocProductId = productAssoc.productId />
</#if>
<#if productAssoc.reason?has_content>
${setRequestAttribute("highlightLabel", productAssoc.reason)}
</#if>
${setRequestAttribute("optProductId", assocProductId)}
${setRequestAttribute("listIndex", listIndex)}
${setRequestAttribute("formNamePrefix", formNamePrefix)}
<#if targetRequestName?has_content>
${setRequestAttribute("targetRequestName", targetRequestName)}
</#if>
${screens.render(productsummaryScreen)}
<#local product = pageProduct />
<#local listIndex = listIndex + 1 />
</#list>
</div>
</div>
${setRequestAttribute("optProductId", "")}
${setRequestAttribute("formNamePrefix", "")}
${setRequestAttribute("targetRequestName", "")}
</#if>
</#macro>
<#assign productValue = product />
<#assign listIndex = 1 />
${setRequestAttribute("productValue", productValue)}
<div id="associated-products">
<#-- also bought -->
<@associated assocProducts=alsoBoughtProducts beforeName="" showName="N"
afterName="${uiLabelMap.ProductAlsoBought}" formNamePrefix="albt" targetRequestName="" />
<#-- obsolete -->
<@associated assocProducts=obsoleteProducts beforeName="" showName="Y" afterName=" ${uiLabelMap.ProductObsolete}"
formNamePrefix="obs" targetRequestName="" />
<#-- cross sell -->
<@associated assocProducts=crossSellProducts beforeName="" showName="N" afterName="${uiLabelMap.ProductCrossSell}"
formNamePrefix="cssl" targetRequestName="crosssell" />
<#-- up sell -->
<@associated assocProducts=upSellProducts beforeName="${uiLabelMap.ProductUpSell} " showName="Y" afterName=":"
formNamePrefix="upsl" targetRequestName="upsell" />
<#-- obsolescence -->
<@associated assocProducts=obsolenscenseProducts beforeName="" showName="Y"
afterName=" ${uiLabelMap.ProductObsolescense}" formNamePrefix="obce" targetRequestName="" />
</div>
<#-- special cross/up-sell area using commonFeatureResultIds (from common feature product search) -->
<#if commonFeatureResultIds?has_content>
<h2>${uiLabelMap.ProductSimilarProducts}</h2>
<div class="productsummary-container">
<div class="row">
<#list commonFeatureResultIds as commonFeatureResultId>
${setRequestAttribute("optProductId", commonFeatureResultId)}
${setRequestAttribute("listIndex", commonFeatureResultId_index)}
${setRequestAttribute("formNamePrefix", "cfeatcssl")}
<#-- ${setRequestAttribute("targetRequestName", targetRequestName)} -->
${screens.render(productsummaryScreen)}
</#list>
</div>
</div>
</#if>
<hr>
<div class="product-tags">
<h4>${uiLabelMap.EcommerceProductTags}</h4>
<#if productTags??>
<p class="titleAddTags"><strong>${uiLabelMap.EcommerceProductTagsDetail}:</strong></p>
<ul>
<li>
<#assign no = 0 />
<#list productTags?keys?sort as productTag>
<#assign tagValue = productTags.get(productTag)!/>
<#if tagValue?has_content>
<span>
<a href="javascript:void(0);" id="productTag_${productTag}">${productTag}</a>
(${tagValue}) <#if no < (productTags.size() - 1)> | </#if>
</span>
<#assign no = no + 1 />
</#if>
</#list>
</li>
</ul>
</#if>
<div class="form-group">
<label>${uiLabelMap.EcommerceAddYourTags}:</label>
<form method="post" action="<@ofbizUrl>addProductTags</@ofbizUrl>" name="addProductTags">
<input type="hidden" name="productId" value="${product.productId!}"/>
<div class="input-group">
<input class="inputProductTags form-control" type="text" value="" name="productTags" id="productTags" size="40"/>
<span class="input-group-btn"><input class="buttonProductTags btn btn-outline-secondary" type="submit" value="${uiLabelMap.EcommerceAddTags}" name="addTag"/></span>
</div>
</form>
</div>
<span>${uiLabelMap.EcommerceAddTagsDetail}</span>
</div>
<form action="<@ofbizUrl>tagsearch</@ofbizUrl>" method="post" name="productTagsearchform" id="productTagsearchform">
<input type="hidden" name="keywordTypeId" value="KWT_TAG"/>
<input type="hidden" name="statusId" value="KW_APPROVED"/>
<input type="hidden" name="clearSearch" value="Y"/>
<input type="hidden" name="VIEW_SIZE" value="10"/>
<input type="hidden" name="PAGING" value="Y"/>
<input type="hidden" name="SEARCH_STRING" id="productTagStr"/>
</form>
</div>
</div>