www.prestigeflowers.co.uk
Open in
urlscan Pro
2606:4700:10::6816:3055
Public Scan
Submitted URL: https://www.hermes.cloudgamingservers.com/
Effective URL: https://www.prestigeflowers.co.uk/charity-flowers/majestic?kk=a4c6294-192fec8cf9e-1bdcd&sfeed&utm_source=kelkoouk&utm_medium=cpc&u...
Submission: On November 06 via automatic, source certstream-suspicious — Scanned from GB
Effective URL: https://www.prestigeflowers.co.uk/charity-flowers/majestic?kk=a4c6294-192fec8cf9e-1bdcd&sfeed&utm_source=kelkoouk&utm_medium=cpc&u...
Submission: On November 06 via automatic, source certstream-suspicious — Scanned from GB
Form analysis
4 forms found in the DOMPOST https://www.prestigeflowers.co.uk/search
<form id="search-form" class="search-form" method="post" action="https://www.prestigeflowers.co.uk/search">
<input type="hidden" name="csrf_token" value="MTczMDg1MTQyNlBadHpWVVd4NnpNQzE4b0hLdDE0OWhyV1V3Nkd2WlN5"> <i class="fa-solid fa-magnifying-glass"></i>
<label class="search-bar" id="instantsearch-box" for="search">
<input type="text" class="placeholder-input" placeholder="Search Prestige Flowers..." style="display: none;">
<input name="search_box" id="search" class="search-input" type="search" placeholder="Search Prestige Flowers..." aria-label="Search"><span hidden="">Loading...</span></label>
</form>
POST https://www.prestigeflowers.co.uk/cart
<form id="form_add_cart" action="https://www.prestigeflowers.co.uk/cart" method="post">
<!-- Addons -->
<div class="product-step-box step2 mandatory">
<div class="product-step">
<span class="stage">
<i class="fa-regular fa-gift"></i>
</span>
<span class="product-step-title"> Add a Little Extra? </span>
</div>
<div class="product-step-box extras-box show_checkbox">
<div class="step-content included-addons">
<div class="included-addon">
<div class="included-addon__img">
<div class="image-wrapper"><img class="" alt="Premium Feed"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/premium-flower-food.webp"> </div>
</div>
<div class="included-addon__desc">
<span class="included-addon__title">Premium Feed</span>
<span class="included-addon__price">FREE!</span>
<label class="checkmark-container" for="attr_addons_1337_Premium Feed">
<span style="font-size: 0px;">Premium Feed</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1337_Premium Feed');" toggle-id="attr_addons_1337_PremiumFeed" checked=""
readonly="" class="included-addon__checkbox" id="attr_addons_1337_Premium Feed" name="attr_addons_1337_Premium Feed">
<span class="checkmark"></span>
</label>
</div>
</div>
</div>
</div>
<div class="product-step-box extras-box show_checkbox">
<div class="step-content featured-addons">
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="180g Pralines"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Chocolate-RedPraline.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Luxury Chocolates 180g</span>
<span class="featured-addon__price">£4.99</span>
<div class="tooltip" data-content="Delicious praline chocolates, perfect for any occasion, may contain nuts*">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1007_180g Pralines">
<span style="font-size: 0px;">Luxury Chocolates 180g</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1007_180g Pralines');"
toggle-id="attr_addons_1007_180gPralines" class="featured-addon__checkbox" id="attr_addons_1007_180g Pralines" name="attr_addons_1007_180g Pralines">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="400g Pralines"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Chocolate-BlackPraline.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Deluxe Chocolates 400g</span>
<span class="featured-addon__price">£9.99</span>
<div class="tooltip" data-content="A box of delicious luxury praline chocolates, may contain nuts*">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1008_400g Pralines">
<span style="font-size: 0px;">Deluxe Chocolates 400g</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1008_400g Pralines');"
toggle-id="attr_addons_1008_400gPralines" class="featured-addon__checkbox" id="attr_addons_1008_400g Pralines" name="attr_addons_1008_400g Pralines">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="Clear Vase"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/new-clear-vase.png"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Clear Vase</span>
<span class="featured-addon__price">£4.99</span>
<div class="tooltip" data-content="A clear and versatile glass vase perfect for this bouquet">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_5_Clear Vase">
<span style="font-size: 0px;">Clear Vase</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_5_Clear Vase');" toggle-id="attr_addons_5_ClearVase"
class="featured-addon__checkbox" id="attr_addons_5_Clear Vase" name="attr_addons_5_Clear Vase">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="PP2202"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/aft-tea-postal-ENJOY.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Afternoon Tea Hamper</span>
<span class="featured-addon__price">£16.99</span>
<div class="tooltip" data-content="Delicious treats hamper perfect for celebrating">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1049_PP2202">
<span style="font-size: 0px;">Afternoon Tea Hamper</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1049_PP2202');" toggle-id="attr_addons_1049_PP2202"
class="featured-addon__checkbox" id="attr_addons_1049_PP2202" name="attr_addons_1049_PP2202">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="Guylian Chocolates"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Chocolate-guylian.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Guylian Chocolates</span>
<span class="featured-addon__price">£2.99</span>
<div class="tooltip" data-content="Delicious Guylian chocolates the perfect accompaniment with flower. Weighs 42g">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1058_Guylian Chocolates">
<span style="font-size: 0px;">Guylian Chocolates</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1058_Guylian Chocolates');"
toggle-id="attr_addons_1058_GuylianChocolates" class="featured-addon__checkbox" id="attr_addons_1058_Guylian Chocolates" name="attr_addons_1058_Guylian Chocolates">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="Birthday Cake"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Gift-BirthdayCake.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Birthday Cake</span>
<span class="featured-addon__price">£4.99</span>
<div class="tooltip" data-content="A delicious hand made birthday cake by Valley Bakery.">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1055_Birthday Cake">
<span style="font-size: 0px;">Birthday Cake</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1055_Birthday Cake');" toggle-id="attr_addons_1055_BirthdayCake"
class="featured-addon__checkbox" id="attr_addons_1055_Birthday Cake" name="attr_addons_1055_Birthday Cake">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt="Cuddly Bear"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Bear.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Luxury Teddy Bear</span>
<span class="featured-addon__price">£9.99</span>
<div class="tooltip" data-content="Limited Edition handmade teddy ready for his new home!
">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1051_Cuddly Bear">
<span style="font-size: 0px;">Luxury Teddy Bear</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1051_Cuddly Bear');" toggle-id="attr_addons_1051_CuddlyBear"
class="featured-addon__checkbox" id="attr_addons_1051_Cuddly Bear" name="attr_addons_1051_Cuddly Bear">
<span class="checkmark"></span>
</label>
</div>
</div>
<div class="featured-addon">
<div class="featured-addon__img">
<div class="image-wrapper"><img class="" alt=" Premium Assorted Belgian Chocolates - Round Box"
src="https://images.prestigeflowers.co.uk/fetch/w_90,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Chocolate-LuxuryRound.jpg"> </div>
</div>
<div class="featured-addon__desc">
<span class="featured-addon__title">Premium Belgian Chocolates</span>
<span class="featured-addon__price">£19.99</span>
<div class="tooltip" data-content="Delicious hand-crafted Belgian chocolates, perfect for elevating any occasion, this product contains alcohol">
<i class="fa-regular fa-circle-info"></i>
</div>
<label class="checkmark-container" for="attr_addons_1072_ Premium Assorted Belgian Chocolates - Round Box">
<span style="font-size: 0px;">Premium Belgian Chocolates</span> <input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1072_ Premium Assorted Belgian Chocolates - Round Box');"
toggle-id="attr_addons_1072_PremiumAssortedBelgianChocolates-RoundBox" class="featured-addon__checkbox" id="attr_addons_1072_ Premium Assorted Belgian Chocolates - Round Box"
name="attr_addons_1072_ Premium Assorted Belgian Chocolates - Round Box">
<span class="checkmark"></span>
</label>
</div>
</div>
</div>
</div>
</div>
<style>
.product-step-box.extras-box:has(.included-addons .checkmark-container[for="attr_addons_1337_Premium Feed"]):not(:has(.included-addon+.included-addon)) {
display: none;
}
.included-addons .included-addon:has(.checkmark-container[for="attr_addons_1337_Premium Feed"]) {
display: none;
}
</style>
<div class="product-step-box extras-box show_checkbox">
<div class="step-content">
<div class="owl_addons_carousel owl-carousel owl-theme owl-loaded owl-drag">
<div class="owl-stage-outer">
<div class="owl-stage" style="transform: translate3d(0px, 0px, 0px); transition: all; width: 3640px;">
<div class="owl-item active" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1159_Just For You Add On Chocolate ">
<div class="tooltip" data-content="Delicious handcrafted Milk Chocolate 'Just For You' Valley Chocolate Bar (60g)">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_JustForYou-60gChocBar_AddOn(600px-300dpi).jpg"
alt="Just For You Add On Chocolate ">
</span>
<span class="add-description"> Just For You Chocolate Bar <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1159_Just For You Add On Chocolate ');" toggle-id="attr_addons_1159_JustForYouAddOnChocolate"
id="attr_addons_1159_Just For You Add On Chocolate " name="attr_addons_1159_Just For You Add On Chocolate ">
<span class="checkmark"></span>
</label></div>
<div class="owl-item active" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1121_Tea And Biscuits">
<div class="tooltip" data-content="Delicious flower shaped biscuits with Yorkshire Tea Bags">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/new tea and biscuits.jpg"
alt="Tea And Biscuits">
</span>
<span class="add-description"> Tea and Biscuits <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £3.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1121_Tea And Biscuits');" toggle-id="attr_addons_1121_TeaAndBiscuits" id="attr_addons_1121_Tea And Biscuits"
name="attr_addons_1121_Tea And Biscuits">
<span class="checkmark"></span>
</label></div>
<div class="owl-item active" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_61_Sparkling French Wine">
<div class="tooltip" data-content="An elegant and refined bottle of bubbly">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Drinks-Pigalle.jpg"
alt="Sparkling French Wine">
</span>
<span class="add-description"> Sparkling French Wine (75cl) <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £19.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_61_Sparkling French Wine');" toggle-id="attr_addons_61_SparklingFrenchWine" id="attr_addons_61_Sparkling French Wine"
name="attr_addons_61_Sparkling French Wine">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1113_Happy Birthday Postal">
<div class="tooltip" data-content="The ultimate Happy Birthday gift with cake, candles & treats!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_PostalHamper-Birthday-Birthday.jpg"
alt="Happy Birthday Postal">
</span>
<span class="add-description"> Happy Birthday Gift Box <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £18.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1113_Happy Birthday Postal');" toggle-id="attr_addons_1113_HappyBirthdayPostal"
id="attr_addons_1113_Happy Birthday Postal" name="attr_addons_1113_Happy Birthday Postal">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1158_Add On Happy Birthday Chocolate ">
<div class="tooltip" data-content="Delicious handcrafted Milk Chocolate 'Happy Birthday' Valley Chocolate Bar (60g)">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_HappyBirthday-60gChocBar_AddOn(600px-300dpi).jpg"
alt="Add On Happy Birthday Chocolate ">
</span>
<span class="add-description"> Birthday Chocolate Bar <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1158_Add On Happy Birthday Chocolate ');" toggle-id="attr_addons_1158_AddOnHappyBirthdayChocolate"
id="attr_addons_1158_Add On Happy Birthday Chocolate " name="attr_addons_1158_Add On Happy Birthday Chocolate ">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1135_Spring Yankee Candle">
<div class="tooltip" data-content="A beautiful Garden Blooms Yankee Candle, for all occasions!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Candle-YankeeGardenBlooms.jpg"
alt="Spring Yankee Candle">
</span>
<span class="add-description"> Garden Blooms Candle <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £9.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1135_Spring Yankee Candle');" toggle-id="attr_addons_1135_SpringYankeeCandle" id="attr_addons_1135_Spring Yankee Candle"
name="attr_addons_1135_Spring Yankee Candle">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1116_Afternoon Tea Gardening Postal">
<div class="tooltip" data-content="A lovely unique gift for nature lovers, artisan cakes, tea & seasonal planting bulbs! ">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/postal addons_0009_PH-Sent-with-love.jpg"
alt="Afternoon Tea Gardening Postal">
</span>
<span class="add-description"> Afternoon Tea Gardening Gift <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £16.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1116_Afternoon Tea Gardening Postal');" toggle-id="attr_addons_1116_AfternoonTeaGardeningPostal"
id="attr_addons_1116_Afternoon Tea Gardening Postal" name="attr_addons_1116_Afternoon Tea Gardening Postal">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1076_Daffodil Bulbs">
<div class="tooltip" data-content="10 Daffodil Bulbs ready for planting!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF--Daffodils-bulb-add-on (1).jpg"
alt="Daffodil Bulbs">
</span>
<span class="add-description"> Daffodil Bulbs <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1076_Daffodil Bulbs');" toggle-id="attr_addons_1076_DaffodilBulbs" id="attr_addons_1076_Daffodil Bulbs"
name="attr_addons_1076_Daffodil Bulbs">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1048_PP2208">
<div class="tooltip" data-content="Cheers to all occasions with this wine gift!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/wine-postal-ENJOY.jpg" alt="PP2208">
</span>
<span class="add-description"> Wine Hamper <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £16.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1048_PP2208');" toggle-id="attr_addons_1048_PP2208" id="attr_addons_1048_PP2208" name="attr_addons_1048_PP2208">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1155_New Champagne 75cl Add On">
<div class="tooltip" data-content="Luxurious Baron De Villeboerg Champagne perfect for celebrating!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Drinks-BaronVilleboerg.jpg"
alt="New Champagne 75cl Add On">
</span>
<span class="add-description"> Champagne 75cl <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £34.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1155_New Champagne 75cl Add On');" toggle-id="attr_addons_1155_NewChampagne75clAddOn"
id="attr_addons_1155_New Champagne 75cl Add On" name="attr_addons_1155_New Champagne 75cl Add On">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1157_GIN + TONIC CANDLE ADD ON">
<div class="tooltip" data-content="A beautiful Gin & Tonic scented candle, for all occasions!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF--Gin-and-Tonic-candle-add-on.jpg"
alt="GIN + TONIC CANDLE ADD ON">
</span>
<span class="add-description"> Gin + Tonic Candle <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £5.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1157_GIN + TONIC CANDLE ADD ON');" toggle-id="attr_addons_1157_GIN+TONICCANDLEADDON"
id="attr_addons_1157_GIN + TONIC CANDLE ADD ON" name="attr_addons_1157_GIN + TONIC CANDLE ADD ON">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_13_White Wine 37.5cl">
<div class="tooltip" data-content="Smooth, aromatic and refreshing French white wine, perfect to say Cheers! ">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Drinks-WhiteWine37cl.jpg"
alt="White Wine 37.5cl">
</span>
<span class="add-description"> White Wine (37.5cl) <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £7.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_13_White Wine 37.5cl');" toggle-id="attr_addons_13_WhiteWine37.5cl" id="attr_addons_13_White Wine 37.5cl"
name="attr_addons_13_White Wine 37.5cl">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1039_Eco Clear Vase">
<div class="tooltip" data-content="New! Eco handmade vase made from recycled glass
">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/luxe-vase.jpg" alt="Eco Clear Vase">
</span>
<span class="add-description"> Eco Clear Vase <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £8.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1039_Eco Clear Vase');" toggle-id="attr_addons_1039_EcoClearVase" id="attr_addons_1039_Eco Clear Vase"
name="attr_addons_1039_Eco Clear Vase">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1089_Dark Chocolate 90g Valley Chocolate ">
<div class="tooltip" data-content="Delicious handcrafted Dark Chocolate and Raspberry Valley Chocolate Bar (90g)">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Chocolate-DarkRaspberryBar.jpg"
alt="Dark Chocolate 90g Valley Chocolate ">
</span>
<span class="add-description"> Dark Chocolate & Raspberry Bar <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1089_Dark Chocolate 90g Valley Chocolate ');" toggle-id="attr_addons_1089_DarkChocolate90gValleyChocolate"
id="attr_addons_1089_Dark Chocolate 90g Valley Chocolate " name="attr_addons_1089_Dark Chocolate 90g Valley Chocolate ">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1017_Birthday Pick">
<div class="tooltip" data-content="Add a delightful pick for extra surprise
">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/birthday_pick.png" alt="Birthday Pick">
</span>
<span class="add-description"> Happy Birthday Pick <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £2.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1017_Birthday Pick');" toggle-id="attr_addons_1017_BirthdayPick" id="attr_addons_1017_Birthday Pick"
name="attr_addons_1017_Birthday Pick">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_14_Red Wine 37.5cl">
<div class="tooltip" data-content="A rich and full bodied delicious French red wine perfect to say Cheers! ">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Drinks-RedWine37cl.jpg"
alt="Red Wine 37.5cl">
</span>
<span class="add-description"> Red Wine (37.5cl) <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £7.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_14_Red Wine 37.5cl');" toggle-id="attr_addons_14_RedWine37.5cl" id="attr_addons_14_Red Wine 37.5cl"
name="attr_addons_14_Red Wine 37.5cl">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1140_Happy Birthday Candle Add On 2024">
<div class="tooltip" data-content="A colourful Happy Birthday candle to make their day extra special!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Candle-HappyBirthday.jpg"
alt="Happy Birthday Candle Add On 2024">
</span>
<span class="add-description"> Happy Birthday Candle <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £2.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1140_Happy Birthday Candle Add On 2024');" toggle-id="attr_addons_1140_HappyBirthdayCandleAddOn2024"
id="attr_addons_1140_Happy Birthday Candle Add On 2024" name="attr_addons_1140_Happy Birthday Candle Add On 2024">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1117_Lullaby Baby Postal">
<div class="tooltip" data-content="The perfect gift to celebrate a new baby with lovely lamb toys!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_PostalHamper-NewBaby-NewBaby.jpg"
alt="Lullaby Baby Postal">
</span>
<span class="add-description"> New Baby Gift <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £24.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1117_Lullaby Baby Postal');" toggle-id="attr_addons_1117_LullabyBabyPostal" id="attr_addons_1117_Lullaby Baby Postal"
name="attr_addons_1117_Lullaby Baby Postal">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1044_PP2204">
<div class="tooltip" data-content="A delightful pamper hamper filled with goodies!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/pamper-hamper-ENJOY.jpg" alt="PP2204">
</span>
<span class="add-description"> Perfect Pamper Gift <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £16.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1044_PP2204');" toggle-id="attr_addons_1044_PP2204" id="attr_addons_1044_PP2204" name="attr_addons_1044_PP2204">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_111_Hand Blown Vase">
<div class="tooltip" data-content="Handmade artisan vase, each unique
">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/hand_blown_vase.png" alt="Hand Blown Vase">
</span>
<span class="add-description"> Hand Blown Vase <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £9.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_111_Hand Blown Vase');" toggle-id="attr_addons_111_HandBlownVase" id="attr_addons_111_Hand Blown Vase"
name="attr_addons_111_Hand Blown Vase">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1154_Lemon Slices - Valley Sweets">
<div class="tooltip" data-content="Delicious fizzy lemon sweet slices by Valley Sweets">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Sweets-FizzyLemon.jpg"
alt="Lemon Slices - Valley Sweets">
</span>
<span class="add-description"> Fizzy Lemon Sweets <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £3.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1154_Lemon Slices - Valley Sweets');" toggle-id="attr_addons_1154_LemonSlices-ValleySweets"
id="attr_addons_1154_Lemon Slices - Valley Sweets" name="attr_addons_1154_Lemon Slices - Valley Sweets">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1167_PP2206">
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/6eb9b3858437161c45549380690e1975.jpg"
alt="PP2206">
</span>
<span class="add-description"> Chocolate Gift Box Gift <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £16.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1167_PP2206');" toggle-id="attr_addons_1167_PP2206" id="attr_addons_1167_PP2206" name="attr_addons_1167_PP2206">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1141_Moet 75cl">
<div class="tooltip" data-content="The perfect extra for special occasions, delightful bottle of Moet 75cl Champagne">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF-MOET-75.CL.jpg" alt="Moet 75cl">
</span>
<span class="add-description"> Moet 75cl <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £65 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1141_Moet 75cl');" toggle-id="attr_addons_1141_Moet75cl" id="attr_addons_1141_Moet 75cl"
name="attr_addons_1141_Moet 75cl">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1046_Scented Occasion Candle">
<div class="tooltip" data-content="Beautifully frangrant candle, florist selected for the occasion
">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/occasion-candle.png"
alt="Scented Occasion Candle">
</span>
<span class="add-description"> Occasion Candle <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1046_Scented Occasion Candle');" toggle-id="attr_addons_1046_ScentedOccasionCandle"
id="attr_addons_1046_Scented Occasion Candle" name="attr_addons_1046_Scented Occasion Candle">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1077_Tulip Bulbs Queen of the Night">
<div class="tooltip" data-content="10 Tulip Bulbs ready for planting!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF-Tulip-bulb-addon (1).jpg"
alt="Tulip Bulbs Queen of the Night">
</span>
<span class="add-description"> Tulip Planting Bulbs <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1077_Tulip Bulbs Queen of the Night');" toggle-id="attr_addons_1077_TulipBulbsQueenoftheNight"
id="attr_addons_1077_Tulip Bulbs Queen of the Night" name="attr_addons_1077_Tulip Bulbs Queen of the Night">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_15_Moet 37.5cl">
<div class="tooltip" data-content="The perfect extra gift to make any occasion that much more extraordinary">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/half-moet.jpg" alt="Moet 37.5cl">
</span>
<span class="add-description"> Moet (37.5cl) <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £39.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_15_Moet 37.5cl');" toggle-id="attr_addons_15_Moet37.5cl" id="attr_addons_15_Moet 37.5cl"
name="attr_addons_15_Moet 37.5cl">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_58_Gin 5cl + Tonic">
<div class="tooltip" data-content="A classic cocktail drink perfect for celebrating ">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_Drinks-GinTonic.jpg"
alt="Gin 5cl + Tonic">
</span>
<span class="add-description"> Gin 5cl + Tonic <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £4.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_58_Gin 5cl + Tonic');" toggle-id="attr_addons_58_Gin5cl+Tonic" id="attr_addons_58_Gin 5cl + Tonic"
name="attr_addons_58_Gin 5cl + Tonic">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1145_PP2202 Happy Birthday Postcard">
<div class="tooltip" data-content="Delectable Afternoon Tea postal gift, perfect for all occasions! ">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PF_(AddOn)_PostalHamper-AfternoonTea-Birthday.jpg"
alt="PP2202 Happy Birthday Postcard">
</span>
<span class="add-description"> Happy Birthday Afternoon Tea <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £16.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1145_PP2202 Happy Birthday Postcard');" toggle-id="attr_addons_1145_PP2202HappyBirthdayPostcard"
id="attr_addons_1145_PP2202 Happy Birthday Postcard" name="attr_addons_1145_PP2202 Happy Birthday Postcard">
<span class="checkmark"></span>
</label></div>
<div class="owl-item" style="width: 125.5px;"><label class="checkmark-container" for="attr_addons_1043_PP2205">
<div class="tooltip" data-content="A beautiful hamper for gin & tonic lovers!">
<i class="fa-regular fa-circle-info"></i>
</div>
<span class="add-img">
<img class="img-responsive center-block lazy"
src="https://images.prestigeflowers.co.uk/fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/https://www.prestigeflowers.co.uk/images/attributes/268204bda128f87ab4c086f233fd022b9a102906/PH-Gin-2024-Enjoy.jpg" alt="PP2205">
</span>
<span class="add-description"> Gin and Tonic Gift <!--<sub>Add on gift</sub>-->
<span class="add-cost"> £16.99 </span>
</span>
<input type="checkbox" onchange="AddonCrossSell($('#attr_messagecard').val()); checkAddon('suggested_attr_addons_1043_PP2205');" toggle-id="attr_addons_1043_PP2205" id="attr_addons_1043_PP2205" name="attr_addons_1043_PP2205">
<span class="checkmark"></span>
</label></div>
</div>
</div>
<div class="owl-nav">
<div class="owl-prev bg-brand disabled"></div>
<div class="owl-next bg-brand"></div>
</div>
<div class="owl-dots">
<div class="owl-dot active"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
<div class="owl-dot"><span></span></div>
</div>
</div>
<script>
$(function() {
var owl = $('.owl_addons_carousel').owlCarousel({
loop: false,
autoplay: false,
dots: true,
nav: true,
items: 3,
slideBy: 5,
lazyLoad: true,
lazyContent: true,
navText: ['', ''],
navClass: ['owl-prev bg-brand', 'owl-next bg-brand'],
dotsClass: 'owl-dots',
responsive: {
480: {
items: 3
},
768: {
items: 3
}
}
});
$(':checkbox[readonly]').on('click change', function() {
return false;
});
});
</script>
</div>
</div>
<div class="clearfix"></div>
<!-- End Addons -->
<div class="product-step-box step3">
<div class="product-step">
<span class="stage">
<i class="fa-regular fa-pen"></i>
</span>
<span class="product-step-title" onclick="$('#attr_messagecard').focus();"> Write a Message Card </span>
</div>
<div class="step-content personalised-card-wrapper ">
<script>
var apiURL = "https://cards.prestigeflowers.co.uk";
var currentURL = "content=charity-flowers&id=majestic&kk=a4c6294-192fec8cf9e-1bdcd&sfeed=&utm_source=kelkoouk&utm_medium=cpc&utm_campaign=kelkooclick&utm_source_platform=KelkooGroup&utm_term=Prestige+Flowers+Majestic";
var editURL =
"content=charity-flowers&id=majestic&kk=a4c6294-192fec8cf9e-1bdcd&sfeed=&utm_source=kelkoouk&utm_medium=cpc&utm_campaign=kelkooclick&utm_source_platform=KelkooGroup&utm_term=Prestige+Flowers+Majestic&action=update_personalised_card";
var useS3 = true;
</script>
<script src="https://www.prestigeflowers.co.uk/styles/main/js/popper.min.js" defer=""></script>
<link rel="preload" as="script" href="https://www.prestigeflowers.co.uk/styles/main/js/bootstrap.min.js">
<script src="https://www.prestigeflowers.co.uk/styles/main/js/bootstrap.min.js" defer=""></script>
<script src="https://www.prestigeflowers.co.uk/styles/main/printzware/js/jquery-3.1.0.min.js"></script>
<script src="https://www.prestigeflowers.co.uk/styles/main/printzware/js/jquery.ba-postmessage.js"></script>
<script src="https://www.prestigeflowers.co.uk/styles/main/printzware/js/printzware.js?v=268204bda128f87ab4c086f233fd022b9a102906"></script>
<script>
function workOutPersonalisedCards(date) {
var today_d = 3;
var today_h = 00;
var today_m = 03;
var today_y = 2024;
var card_cutoff_h = 18;
var card_cutoff_m = 0;
// Saturday cuts off at 4pm
if (today_d === 6) {
card_cutoff_h = 16;
card_cutoff_m = 0;
}
var cutoff = false;
var tomorrow_d = 07;
var today_day = 06;
// Only check time cutoffs if it's next day
if (date.getDate() === tomorrow_d) {
if (today_h == card_cutoff_h && (today_m >= card_cutoff_m && today_m <= 59)) {
cutoff = true;
}
if (today_h > card_cutoff_h) {
cutoff = true;
}
}
// Sunday cuts off at 2pm (peak only)
if (today_d === 0 && date.getDate() === tomorrow_d) {
cutoff = true; // Always cut off on Sunday
}
var is_unavailable_day = ((today_d === 6 && cutoff === true) || (today_d === 0 && cutoff === true));
// Do our date cut-off by special date.
var month = date.getUTCMonth() + 1; //months from 1-12
var day = date.getUTCDate();
var year = date.getUTCFullYear();
// Need to get it off?
if ((date.getDate() === (tomorrow_d) && cutoff === true) || is_unavailable_day === true) {
$('.add_personalised_message').fadeOut();
$('.message-card-standard').fadeIn();
$("#standard").prop("checked", true);
} else {
$('.add_personalised_message').fadeIn();
}
}
</script>
<script>
jQuery(document).ready(function() {
function selectStandardCard() {
jQuery('.message-card-standard').removeClass('hidden');
jQuery('.message-card-standard').fadeIn();
jQuery('#standard').prop("checked", true);
}
jQuery('.standard_card').click(selectStandardCard);
jQuery('#attr_messagecard').focus(selectStandardCard);
jQuery('.add_personalised_message').click(function() {
jQuery('#standard').prop("checked", false);
jQuery("#personalised").prop("checked", true);
jQuery('.message-card-standard').hide();
var check = $('#cards .pw-card-image a');
if (check.length) {
// Open the edit url
opendialog(check.data('edit-url'));
} else {
// New order
opendialog(
'https://cards.prestigeflowers.co.uk/open.php?h=69a31be016d684184e13825e7c687b6473d01aca8192f5084dfbfbd234cbaee4&c=485&t=card&f=open&m=&o=pw672ab26332f8b&tag=0&clientCat=&p=&ts=1730851427&url=https%3A%2F%2Fwww.prestigeflowers.co.uk'
);
}
var delivery_date_selected = $("#delivery_date").datepicker('getFormattedDate');
getDeliveryOptions(delivery_date_selected, '5709');
});
jQuery("#delivery_date").on("changeDate", function(event) {
workOutPersonalisedCards(event.date);
});
jQuery('#pwCancelButton').click(function() {
console.log("Close Modal");
jQuery('#pw_modal').animate({
display: "none",
}, 400);
});
workOutPersonalisedCards($('#delivery_date').datepicker('getDate'));
});
function removeMessageCard() {
jQuery('.standard_card').hide();
jQuery('.message-card-seperator').hide();
};
</script>
<div class="col-xs-12" style="margin-bottom: 20px;">
<label for="personalised" class="personalised-card-container add_personalised_message">
<div class="left-section">
<div class="checkmark-container">
<input class="non-pretty ignore" type="radio" name="attr_card_type" id="personalised" value="personalised">
<p class="personalised-card-option"> Add a Personalised Card </p>
</div>
<div class="personalised-card-message">
<ul>
<li>Luxury hand finished cards</li>
<li>Cards for all occasions</li>
<li>Choose from many designs</li>
<li>Easily upload your pictures</li>
</ul>
<span class="personalised-card-price"> + £3.99 <s>£5.99</s> </span>
</div>
</div>
<div class="right-section">
<div class="personalised-card-preview">
<div id="cards"></div>
<!-- Modal -->
<div class="modal fade" id="pw_modal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true" data-bs-backdrop="static">
<div class="modal-dialog modal-dialog-centered">
<div class="modal-content">
<div class="modal-body">
<iframe id="pw_iframe" style="border: 0px;" src="about:blank" width="100%" height="100%"></iframe>
</div>
</div>
</div>
</div>
<div class="preview_container">
<img id="personalised_example" width="100" height="140"
src=""
alt="">
<!-- <div class="personalise-button">Personalise</div> -->
</div>
</div>
</div>
</label>
</div><!--col-xs-12-->
<section class="message-card-section">
<label class="message-card-trigger " onclick="OpenVideo();">
<input class="ignore" type="checkbox" name="recorded_a_message" value="yes">
<!-- <input class="ignore" checked="checked" type="radio" name="attr_card_type" id="standard" value="standard" /> -->
<div class="checkmark-container">
<div class="checkmark"></div>
<p>Send a FREE recorded message</p>
</div>
<div class="how-it-works">
<div class="step">
<img src="https://images.prestigeflowers.co.uk/fetch/w_120/https://www.prestigeflowers.co.uk/images/video-message/record.jpg" alt="step 1">
<span>Record your message</span>
</div>
<div class="step">
<img src="https://images.prestigeflowers.co.uk/fetch/w_120/https://www.prestigeflowers.co.uk/images/video-message/send.jpg" alt="step 2">
<span>We'll send with your gift</span>
</div>
</div>
<button type="button" class="btn"> Record Now! </button>
</label>
<div class="message-card-record-container hidden" id="voice-recording-container">
<div class="message-card-wrapper">
<div class="message-card-record">
<div class="close-option-top">
<span class="btn" id="close-window-option" onclick="CloseWindow();">
<i class="fa fal fa-xmark"></i>
</span>
</div>
<div class="heading">
<div class="heading-track" id="heading-track">
<div class="video-message recording">
<div id="video-preview">
<div class="message-recording-preview" id="recording_container">
<div class="recording-preview hidden"></div>
<div class="recording-preview-text hidden"></div>
<video controls="" autoplay="" playsinline="" id="recording_element" style="width: 100%;" class="hidden"></video>
</div>
</div>
<div class="content">
<div class="recording"></div>
</div>
</div>
<div class="intro">
<img src="images/video-message/family-selfie-min-new.jpg" alt="step 1">
<div class="content">
<h1>Send your FREE video message with every gift</h1>
</div>
</div>
<div class="voice-message recording">
<div class="content">
<div class="visualiser">
<div class="bar"></div>
<div class="bar"></div>
<div class="bar"></div>
<div class="bar"></div>
<div class="bar"></div>
</div>
<div id="voice-preview"></div>
<h1>Record a voice message</h1>
<p> Just <b>press the 'Record' button</b> to start recording your voice message, you will have up to 30 seconds.</p>
</div>
</div>
</div>
</div>
<div class="choices">
<div class="video-error hidden">
<h3 style="color: red;">Please allow access to your camera and microphone to start recording.</h3>
</div>
<div class="voice-error hidden">
<h3 style="color: red;">Please allow access to your microphone to start recording.</h3>
</div>
<div class="choices-heading">
<h3 class="start">Let's Get Started</h3>
<h3 class="finish hidden">Happy With Your Recording?</h3>
</div>
<div class="options-wrapper">
<div class="options-track" id="options-track">
<div class="video-recording">
<div class="back-option">
<span class="btn back-button" onclick="VoiceCancel()">
<i class="fa fal fa-chevron-left"></i>
</span>
<text>Back</text>
</div>
<div class="send-video-option hidden" onclick="CloseVideo(false, true);">
<span class="btn send-video-button" id="send-video">
<i class="fa fal fa-send"></i>
</span>
<text>Add To Order!</text>
</div>
<div class="completed-video-option hidden" onclick="ConfirmVideo();">
<span class="btn complete-video-button" id="complete_video">
<i class="fa fal fa-check"></i>
</span>
<text>Save & Review</text>
</div>
<div class="video-option">
<span class="btn voice-button" id="record_video_audio">
<i class="fa fal fa-video-camera"></i>
</span>
<text>Record</text>
</div>
<div id="video-finish-option"></div>
</div>
<div class="intro">
<div class="video-option">
<span class="btn voice-button" onclick="VideoStart();">
<i class="fa fal fa-video-camera"></i>
</span>
<text>Video Message</text>
</div>
<div class="voice-option">
<span class="btn voice-button" onclick="VoiceStart();">
<i class="fa fal fa-microphone"></i>
</span>
<text>Voice Message</text>
</div>
<div class="close-option">
<span class="btn" onclick="CloseVideo();">
<i class="fa fal fa-xmark"></i>
</span>
<text>Close</text>
</div>
</div>
<div class="voice-recording">
<div class="back-option">
<span class="btn back-button" onclick="VoiceCancel();">
<i class="fa fal fa-chevron-left"></i>
</span>
<text>Back</text>
</div>
<div class="voice-option">
<span class="btn voice-button" id="record_audio">
<i class="fa fal fa-microphone"></i>
</span>
<text>Record</text>
</div>
<div class="completed-voice-option hidden" onclick="CloseVideo(false, true);">
<span class="btn complete-voice-button" id="complete_voice">
<i class="fa fal fa-check"></i>
</span>
<text>Save & Send</text>
</div>
<div id="voice-finish-option"></div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</section>
<div id="stop-option" class="hidden">
<span class="btn stop-button" id="record_stop">
<i class="fa fal fa-stop"></i>
</span> Stop Recording
</div>
<script>
var quiet = false;
var old_trigger_content;
var video = document.querySelector("#recording_element");
var element = document.getElementById('recording_element');
element.muted = "muted";
function VideoStart() {
console.log("Video Start");
$('#heading-track').css("left", "0");
$('#options-track').css("left", "0");
$('#recording_container').appendTo('#video-preview');
$('#close-window-option').attr("onclick", "CloseRecording();");
$('#recording_container').children("video").removeAttr('controls');
$('#record_audio').children("video").removeClass('controls');
$('#recording_element').removeAttr("controls");
$('.recording-preview-text').removeClass("hidden");
element.muted = "muted";
if (navigator.mediaDevices.getUserMedia) {
navigator.mediaDevices.getUserMedia({
video: true,
audio: true
}).then(function(stream) {
video.srcObject = stream;
}).catch(function(err0r) {
console.log("Something went wrong!");
});
}
}
function VoiceStart() {
console.log("VoiceStart");
$('.visualiser').removeClass("completed").removeClass("active");
$('#heading-track').css("left", "-200%");
$('#options-track').css("left", "-200%");
}
function VoiceCancel() {
$('#heading-track').css("left", "-100%");
$('#options-track').css("left", "-100%");
$('#recording_element').attr("src", "");
$('.visualiser').removeClass("completed").removeClass("active");
$('#recording_element').addClass("hidden");
$('.video-option').removeClass("hidden");
$('.finish').addClass("hidden");
$('.start').removeClass("hidden");
$('.send-video-option').addClass("hidden");
$('.message-card-record').removeClass("preview");
$('.video-option').children("text").text("Video Message");
$('.voice-option').children("text").text("Voice Message");
$('.completed-voice-option').addClass("hidden");
$('.completed-video-option').addClass("hidden");
$('#attr_videomsg').val("");
}
function OpenVideo() {
$('#voice-recording-container').removeClass("hidden");
}
function ConfirmVideo() {
$('.message-card-record').addClass("preview");
$('.send-video-option').removeClass("hidden");
$('.completed-video-option').addClass("hidden");
$('.start').addClass("hidden");
$(".finish").removeClass("hidden");
$('.video-option').addClass("hidden");
}
function CloseVideo(quiet = true, success = false) {
$('#voice-recording-container').addClass("hidden");
quiet = quiet;
if (success) {
old_trigger_content = $('.message-card-trigger .how-it-works').html();
$('.message-card-trigger .how-it-works').html("");
$('#recording_element').appendTo('.message-card-trigger .how-it-works');
$('.message-card-trigger').attr("onClick", "");
$('.message-card-trigger .btn').text("Start Again!").click(function() {
console.log("click");
$('.message-card-trigger .btn').text("Get Started");
$('.message-card-trigger .btn').unbind('click');
$('.message-card-trigger').attr("onClick", "OpenVideo();");
$('#recording_element').appendTo('#video-preview');
$('.message-card-trigger .how-it-works').html(old_trigger_content);
VoiceCancel();
OpenVideo();
});
}
// AwaitUpload();
}
function CloseWindow() {
$('#voice-recording-container').addClass("hidden");
quiet = quiet;
};
function CloseRecording() {
$('#voice-recording-container').addClass("hidden");
$('#heading-track').css("left", "-100%");
$('#options-track').css("left", "-100%");
};
function CloseStartedRecording() {
$('#voice-recording-container').addClass("hidden");
quiet = quiet;
recorder.stream.getTracks().forEach(function(track) {
track.stop();
});
$('#heading-track').css("left", "-100%");
$('#options-track').css("left", "-100%");
$('#recording_element').attr("src", "");
$('.visualiser').removeClass("completed").removeClass("active");
$('#recording_element').addClass("hidden");
$('.video-option').removeClass("hidden");
$('.finish').addClass("hidden");
$('.start').removeClass("hidden");
$('.send-video-option').addClass("hidden");
$('.message-card-record').removeClass("preview");
$('.recording-preview').addClass("hidden");
$('.video-option').children("text").text("Video Message");
$('.voice-option').children("text").text("Voice Message");
$('.completed-voice-option').addClass("hidden");
$('.completed-video-option').addClass("hidden");
$('.video-message').removeClass("active");
$('#stop-option').addClass('hidden');
$('#record_video_audio').removeClass('hidden');
$('#record_audio').removeClass('hidden');
$('.video-option').removeClass("hidden");
$('.voice-option').removeClass("hidden");
$('.back-option').removeClass("hidden");
$('#attr_videomsg').val("");
};
function AwaitUpload() {
$('#submit').attr("disabled", "true");
$('#submit').addClass("disabled");
$('#submit').text("Uploading...");
}
</script>
<style>
.hidden {
display: none;
}
</style>
<script>
'use strict';
// Last time updated: 2021-03-09 3:20:23 AM UTC
// ________________
// RecordRTC v5.6.2
// Open-Sourced: https://github.com/muaz-khan/RecordRTC
// --------------------------------------------------
// Muaz Khan - www.MuazKhan.com
// MIT License - www.WebRTC-Experiment.com/licence
// --------------------------------------------------
"use strict";
function RecordRTC(mediaStream, config) {
function startRecording(config2) {
return config.disableLogs || console.log("RecordRTC version: ", self.version), config2 && (config = new RecordRTCConfiguration(mediaStream, config2)), config.disableLogs || console.log("started recording " + config.type + " stream."),
mediaRecorder ? (mediaRecorder.clearRecordedData(), mediaRecorder.record(), setState("recording"), self.recordingDuration && handleRecordingDuration(), self) : (initRecorder(function() {
self.recordingDuration && handleRecordingDuration()
}), self)
}
function initRecorder(initCallback) {
initCallback && (config.initCallback = function() {
initCallback(), initCallback = config.initCallback = null
});
var Recorder = new GetRecorderType(mediaStream, config);
mediaRecorder = new Recorder(mediaStream, config), mediaRecorder.record(), setState("recording"), config.disableLogs || console.log("Initialized recorderType:", mediaRecorder.constructor.name, "for output-type:", config.type)
}
function stopRecording(callback) {
function _callback(__blob) {
if (!mediaRecorder) return void("function" == typeof callback.call ? callback.call(self, "") : callback(""));
Object.keys(mediaRecorder).forEach(function(key) {
"function" != typeof mediaRecorder[key] && (self[key] = mediaRecorder[key])
});
var blob = mediaRecorder.blob;
if (!blob) {
if (!__blob) throw "Recording failed.";
mediaRecorder.blob = blob = __blob
}
if (blob && !config.disableLogs && console.log(blob.type, "->", bytesToSize(blob.size)), callback) {
var url;
try {
url = URL.createObjectURL(blob)
} catch (e) {}
"function" == typeof callback.call ? callback.call(self, url) : callback(url)
}
config.autoWriteToDisk && getDataURL(function(dataURL) {
var parameter = {};
parameter[config.type + "Blob"] = dataURL, DiskStorage.Store(parameter)
})
}
return callback = callback || function() {}, mediaRecorder ? "paused" === self.state ? (self.resumeRecording(), void setTimeout(function() {
stopRecording(callback)
}, 1)) : ("recording" === self.state || config.disableLogs || console.warn('Recording state should be: "recording", however current state is: ', self.state), config.disableLogs || console.log("Stopped recording " + config.type +
" stream."), "gif" !== config.type ? mediaRecorder.stop(_callback) : (mediaRecorder.stop(), _callback()), void setState("stopped")) : void warningLog()
}
function pauseRecording() {
return mediaRecorder ? "recording" !== self.state ? void(config.disableLogs || console.warn("Unable to pause the recording. Recording state: ", self.state)) : (setState("paused"), mediaRecorder.pause(), void(config.disableLogs || console
.log("Paused recording."))) : void warningLog()
}
function resumeRecording() {
return mediaRecorder ? "paused" !== self.state ? void(config.disableLogs || console.warn("Unable to resume the recording. Recording state: ", self.state)) : (setState("recording"), mediaRecorder.resume(), void(config.disableLogs ||
console.log("Resumed recording."))) : void warningLog()
}
function readFile(_blob) {
postMessage((new FileReaderSync).readAsDataURL(_blob))
}
function getDataURL(callback, _mediaRecorder) {
function processInWebWorker(_function) {
try {
var blob = URL.createObjectURL(new Blob([_function.toString(), "this.onmessage = function (eee) {" + _function.name + "(eee.data);}"], {
type: "application/javascript"
})),
worker = new Worker(blob);
return URL.revokeObjectURL(blob), worker
} catch (e) {}
}
if (!callback) throw "Pass a callback function over getDataURL.";
var blob = _mediaRecorder ? _mediaRecorder.blob : (mediaRecorder || {}).blob;
if (!blob) return config.disableLogs || console.warn("Blob encoder did not finish its job yet."), void setTimeout(function() {
getDataURL(callback, _mediaRecorder)
}, 1e3);
if ("undefined" == typeof Worker || navigator.mozGetUserMedia) {
var reader = new FileReader;
reader.readAsDataURL(blob), reader.onload = function(event) {
callback(event.target.result)
}
} else {
var webWorker = processInWebWorker(readFile);
webWorker.onmessage = function(event) {
callback(event.data)
}, webWorker.postMessage(blob)
}
}
function handleRecordingDuration(counter) {
if (counter = counter || 0, "paused" === self.state) return void setTimeout(function() {
handleRecordingDuration(counter)
}, 1e3);
if ("stopped" !== self.state) {
if (counter >= self.recordingDuration) return void stopRecording(self.onRecordingStopped);
counter += 1e3, setTimeout(function() {
handleRecordingDuration(counter)
}, 1e3)
}
}
function setState(state) {
self && (self.state = state, "function" == typeof self.onStateChanged.call ? self.onStateChanged.call(self, state) : self.onStateChanged(state))
}
function warningLog() {
config.disableLogs !== !0 && console.warn(WARNING)
}
if (!mediaStream) throw "First parameter is required.";
config = config || {
type: "video"
}, config = new RecordRTCConfiguration(mediaStream, config);
var mediaRecorder, self = this,
WARNING = 'It seems that recorder is destroyed or "startRecording" is not invoked for ' + config.type + " recorder.",
returnObject = {
startRecording: startRecording,
stopRecording: stopRecording,
pauseRecording: pauseRecording,
resumeRecording: resumeRecording,
initRecorder: initRecorder,
setRecordingDuration: function(recordingDuration, callback) {
if ("undefined" == typeof recordingDuration) throw "recordingDuration is required.";
if ("number" != typeof recordingDuration) throw "recordingDuration must be a number.";
return self.recordingDuration = recordingDuration, self.onRecordingStopped = callback || function() {}, {
onRecordingStopped: function(callback) {
self.onRecordingStopped = callback
}
}
},
clearRecordedData: function() {
return mediaRecorder ? (mediaRecorder.clearRecordedData(), void(config.disableLogs || console.log("Cleared old recorded data."))) : void warningLog()
},
getBlob: function() {
return mediaRecorder ? mediaRecorder.blob : void warningLog()
},
getDataURL: getDataURL,
toURL: function() {
return mediaRecorder ? URL.createObjectURL(mediaRecorder.blob) : void warningLog()
},
getInternalRecorder: function() {
return mediaRecorder
},
save: function(fileName) {
return mediaRecorder ? void invokeSaveAsDialog(mediaRecorder.blob, fileName) : void warningLog()
},
getFromDisk: function(callback) {
return mediaRecorder ? void RecordRTC.getFromDisk(config.type, callback) : void warningLog()
},
setAdvertisementArray: function(arrayOfWebPImages) {
config.advertisement = [];
for (var length = arrayOfWebPImages.length, i = 0; i < length; i++) config.advertisement.push({
duration: i,
image: arrayOfWebPImages[i]
})
},
blob: null,
bufferSize: 0,
sampleRate: 0,
buffer: null,
reset: function() {
"recording" !== self.state || config.disableLogs || console.warn("Stop an active recorder."), mediaRecorder && "function" == typeof mediaRecorder.clearRecordedData && mediaRecorder.clearRecordedData(), mediaRecorder = null,
setState("inactive"), self.blob = null
},
onStateChanged: function(state) {
config.disableLogs || console.log("Recorder state changed:", state)
},
state: "inactive",
getState: function() {
return self.state
},
destroy: function() {
var disableLogsCache = config.disableLogs;
config = {
disableLogs: !0
}, self.reset(), setState("destroyed"), returnObject = self = null, Storage.AudioContextConstructor && (Storage.AudioContextConstructor.close(), Storage.AudioContextConstructor = null), config.disableLogs = disableLogsCache,
config.disableLogs || console.log("RecordRTC is destroyed.")
},
version: "5.6.2"
};
if (!this) return self = returnObject, returnObject;
for (var prop in returnObject) this[prop] = returnObject[prop];
return self = this, returnObject
}
function RecordRTCConfiguration(mediaStream, config) {
return config.recorderType || config.type || (config.audio && config.video ? config.type = "video" : config.audio && !config.video && (config.type = "audio")), config.recorderType && !config.type && (config.recorderType ===
WhammyRecorder || config.recorderType === CanvasRecorder || "undefined" != typeof WebAssemblyRecorder && config.recorderType === WebAssemblyRecorder ? config.type = "video" : config.recorderType === GifRecorder ? config.type = "gif" :
config.recorderType === StereoAudioRecorder ? config.type = "audio" : config.recorderType === MediaStreamRecorder && (getTracks(mediaStream, "audio").length && getTracks(mediaStream, "video").length ? config.type = "video" : !getTracks(
mediaStream, "audio").length && getTracks(mediaStream, "video").length ? config.type = "video" : getTracks(mediaStream, "audio").length && !getTracks(mediaStream, "video").length && (config.type = "audio"))), "undefined" !=
typeof MediaStreamRecorder && "undefined" != typeof MediaRecorder && "requestData" in MediaRecorder.prototype && (config.mimeType || (config.mimeType = "video/webm"), config.type || (config.type = config.mimeType.split("/")[0]), !config
.bitsPerSecond), config.type || (config.mimeType && (config.type = config.mimeType.split("/")[0]), config.type || (config.type = "audio")), config
}
function GetRecorderType(mediaStream, config) {
var recorder;
return (isChrome || isEdge || isOpera) && (recorder = StereoAudioRecorder), "undefined" != typeof MediaRecorder && "requestData" in MediaRecorder.prototype && !isChrome && (recorder = MediaStreamRecorder), "video" === config.type && (
isChrome || isOpera) && (recorder = WhammyRecorder, "undefined" != typeof WebAssemblyRecorder && "undefined" != typeof ReadableStream && (recorder = WebAssemblyRecorder)), "gif" === config.type && (recorder = GifRecorder), "canvas" ===
config.type && (recorder = CanvasRecorder), isMediaRecorderCompatible() && recorder !== CanvasRecorder && recorder !== GifRecorder && "undefined" != typeof MediaRecorder && "requestData" in MediaRecorder.prototype && (getTracks(
mediaStream, "video").length || getTracks(mediaStream, "audio").length) && ("audio" === config.type ? "function" == typeof MediaRecorder.isTypeSupported && MediaRecorder.isTypeSupported("audio/webm") && (recorder =
MediaStreamRecorder) : "function" == typeof MediaRecorder.isTypeSupported && MediaRecorder.isTypeSupported("video/webm") && (recorder = MediaStreamRecorder)), mediaStream instanceof Array && mediaStream.length && (recorder =
MultiStreamRecorder), config.recorderType && (recorder = config.recorderType), !config.disableLogs && recorder && recorder.name && console.log("Using recorderType:", recorder.name || recorder.constructor.name), !recorder && isSafari &&
(recorder = MediaStreamRecorder), recorder
}
function MRecordRTC(mediaStream) {
this.addStream = function(_mediaStream) {
_mediaStream && (mediaStream = _mediaStream)
}, this.mediaType = {
audio: !0,
video: !0
}, this.startRecording = function() {
var recorderType, mediaType = this.mediaType,
mimeType = this.mimeType || {
audio: null,
video: null,
gif: null
};
if ("function" != typeof mediaType.audio && isMediaRecorderCompatible() && !getTracks(mediaStream, "audio").length && (mediaType.audio = !1), "function" != typeof mediaType.video && isMediaRecorderCompatible() && !getTracks(mediaStream,
"video").length && (mediaType.video = !1), "function" != typeof mediaType.gif && isMediaRecorderCompatible() && !getTracks(mediaStream, "video").length && (mediaType.gif = !1), !mediaType.audio && !mediaType.video && !mediaType.gif)
throw "MediaStream must have either audio or video tracks.";
if (mediaType.audio && (recorderType = null, "function" == typeof mediaType.audio && (recorderType = mediaType.audio), this.audioRecorder = new RecordRTC(mediaStream, {
type: "audio",
bufferSize: this.bufferSize,
sampleRate: this.sampleRate,
numberOfAudioChannels: this.numberOfAudioChannels || 2,
disableLogs: this.disableLogs,
recorderType: recorderType,
mimeType: mimeType.audio,
timeSlice: this.timeSlice,
onTimeStamp: this.onTimeStamp
}), mediaType.video || this.audioRecorder.startRecording()), mediaType.video) {
recorderType = null, "function" == typeof mediaType.video && (recorderType = mediaType.video);
var newStream = mediaStream;
if (isMediaRecorderCompatible() && mediaType.audio && "function" == typeof mediaType.audio) {
var videoTrack = getTracks(mediaStream, "video")[0];
isFirefox ? (newStream = new MediaStream, newStream.addTrack(videoTrack), recorderType && recorderType === WhammyRecorder && (recorderType = MediaStreamRecorder)) : (newStream = new MediaStream, newStream.addTrack(videoTrack))
}
this.videoRecorder = new RecordRTC(newStream, {
type: "video",
video: this.video,
canvas: this.canvas,
frameInterval: this.frameInterval || 10,
disableLogs: this.disableLogs,
recorderType: recorderType,
mimeType: mimeType.video,
timeSlice: this.timeSlice,
onTimeStamp: this.onTimeStamp,
workerPath: this.workerPath,
webAssemblyPath: this.webAssemblyPath,
frameRate: this.frameRate,
bitrate: this.bitrate
}), mediaType.audio || this.videoRecorder.startRecording()
}
if (mediaType.audio && mediaType.video) {
var self = this,
isSingleRecorder = isMediaRecorderCompatible() === !0;
mediaType.audio instanceof StereoAudioRecorder && mediaType.video ? isSingleRecorder = !1 : mediaType.audio !== !0 && mediaType.video !== !0 && mediaType.audio !== mediaType.video && (isSingleRecorder = !1), isSingleRecorder === !0 ?
(self.audioRecorder = null, self.videoRecorder.startRecording()) : self.videoRecorder.initRecorder(function() {
self.audioRecorder.initRecorder(function() {
self.videoRecorder.startRecording(), self.audioRecorder.startRecording()
})
})
}
mediaType.gif && (recorderType = null, "function" == typeof mediaType.gif && (recorderType = mediaType.gif), this.gifRecorder = new RecordRTC(mediaStream, {
type: "gif",
frameRate: this.frameRate || 200,
quality: this.quality || 10,
disableLogs: this.disableLogs,
recorderType: recorderType,
mimeType: mimeType.gif
}), this.gifRecorder.startRecording())
}, this.stopRecording = function(callback) {
callback = callback || function() {}, this.audioRecorder && this.audioRecorder.stopRecording(function(blobURL) {
callback(blobURL, "audio")
}), this.videoRecorder && this.videoRecorder.stopRecording(function(blobURL) {
callback(blobURL, "video")
}), this.gifRecorder && this.gifRecorder.stopRecording(function(blobURL) {
callback(blobURL, "gif")
})
}, this.pauseRecording = function() {
this.audioRecorder && this.audioRecorder.pauseRecording(), this.videoRecorder && this.videoRecorder.pauseRecording(), this.gifRecorder && this.gifRecorder.pauseRecording()
}, this.resumeRecording = function() {
this.audioRecorder && this.audioRecorder.resumeRecording(), this.videoRecorder && this.videoRecorder.resumeRecording(), this.gifRecorder && this.gifRecorder.resumeRecording()
}, this.getBlob = function(callback) {
var output = {};
return this.audioRecorder && (output.audio = this.audioRecorder.getBlob()), this.videoRecorder && (output.video = this.videoRecorder.getBlob()), this.gifRecorder && (output.gif = this.gifRecorder.getBlob()), callback && callback(
output), output
}, this.destroy = function() {
this.audioRecorder && (this.audioRecorder.destroy(), this.audioRecorder = null), this.videoRecorder && (this.videoRecorder.destroy(), this.videoRecorder = null), this.gifRecorder && (this.gifRecorder.destroy(), this.gifRecorder = null)
}, this.getDataURL = function(callback) {
function getDataURL(blob, callback00) {
if ("undefined" != typeof Worker) {
var webWorker = processInWebWorker(function(_blob) {
postMessage((new FileReaderSync).readAsDataURL(_blob))
});
webWorker.onmessage = function(event) {
callback00(event.data)
}, webWorker.postMessage(blob)
} else {
var reader = new FileReader;
reader.readAsDataURL(blob), reader.onload = function(event) {
callback00(event.target.result)
}
}
}
function processInWebWorker(_function) {
var url, blob = URL.createObjectURL(new Blob([_function.toString(), "this.onmessage = function (eee) {" + _function.name + "(eee.data);}"], {
type: "application/javascript"
})),
worker = new Worker(blob);
if ("undefined" != typeof URL) url = URL;
else {
if ("undefined" == typeof webkitURL) throw "Neither URL nor webkitURL detected.";
url = webkitURL
}
return url.revokeObjectURL(blob), worker
}
this.getBlob(function(blob) {
blob.audio && blob.video ? getDataURL(blob.audio, function(_audioDataURL) {
getDataURL(blob.video, function(_videoDataURL) {
callback({
audio: _audioDataURL,
video: _videoDataURL
})
})
}) : blob.audio ? getDataURL(blob.audio, function(_audioDataURL) {
callback({
audio: _audioDataURL
})
}) : blob.video && getDataURL(blob.video, function(_videoDataURL) {
callback({
video: _videoDataURL
})
})
})
}, this.writeToDisk = function() {
RecordRTC.writeToDisk({
audio: this.audioRecorder,
video: this.videoRecorder,
gif: this.gifRecorder
})
}, this.save = function(args) {
args = args || {
audio: !0,
video: !0,
gif: !0
}, args.audio && this.audioRecorder && this.audioRecorder.save("string" == typeof args.audio ? args.audio : ""), args.video && this.videoRecorder && this.videoRecorder.save("string" == typeof args.video ? args.video : ""), args.gif &&
this.gifRecorder && this.gifRecorder.save("string" == typeof args.gif ? args.gif : "")
}
}
function bytesToSize(bytes) {
var k = 1e3,
sizes = ["Bytes", "KB", "MB", "GB", "TB"];
if (0 === bytes) return "0 Bytes";
var i = parseInt(Math.floor(Math.log(bytes) / Math.log(k)), 10);
return (bytes / Math.pow(k, i)).toPrecision(3) + " " + sizes[i]
}
function invokeSaveAsDialog(file, fileName) {
if (!file) throw "Blob object is required.";
if (!file.type) try {
file.type = "video/webm"
} catch (e) {}
var fileExtension = (file.type || "video/webm").split("/")[1];
if (fileExtension.indexOf(";") !== -1 && (fileExtension = fileExtension.split(";")[0]), fileName && fileName.indexOf(".") !== -1) {
var splitted = fileName.split(".");
fileName = splitted[0], fileExtension = splitted[1]
}
var fileFullName = (fileName || Math.round(9999999999 * Math.random()) + 888888888) + "." + fileExtension;
if ("undefined" != typeof navigator.msSaveOrOpenBlob) return navigator.msSaveOrOpenBlob(file, fileFullName);
if ("undefined" != typeof navigator.msSaveBlob) return navigator.msSaveBlob(file, fileFullName);
var hyperlink = document.createElement("a");
hyperlink.href = URL.createObjectURL(file), hyperlink.download = fileFullName, hyperlink.style = "display:none;opacity:0;color:transparent;", (document.body || document.documentElement).appendChild(hyperlink), "function" == typeof hyperlink
.click ? hyperlink.click() : (hyperlink.target = "_blank", hyperlink.dispatchEvent(new MouseEvent("click", {
view: window,
bubbles: !0,
cancelable: !0
}))), URL.revokeObjectURL(hyperlink.href)
}
function isElectron() {
return "undefined" != typeof window && "object" == typeof window.process && "renderer" === window.process.type || (!("undefined" == typeof process || "object" != typeof process.versions || !process.versions.electron) || "object" ==
typeof navigator && "string" == typeof navigator.userAgent && navigator.userAgent.indexOf("Electron") >= 0)
}
function getTracks(stream, kind) {
return stream && stream.getTracks ? stream.getTracks().filter(function(t) {
return t.kind === (kind || "audio")
}) : []
}
function setSrcObject(stream, element) {
"srcObject" in element ? element.srcObject = stream : "mozSrcObject" in element ? element.mozSrcObject = stream : element.srcObject = stream
}
function getSeekableBlob(inputBlob, callback) {
if ("undefined" == typeof EBML) throw new Error("Please link: https://www.webrtc-experiment.com/EBML.js");
var reader = new EBML.Reader,
decoder = new EBML.Decoder,
tools = EBML.tools,
fileReader = new FileReader;
fileReader.onload = function(e) {
var ebmlElms = decoder.decode(this.result);
ebmlElms.forEach(function(element) {
reader.read(element)
}), reader.stop();
var refinedMetadataBuf = tools.makeMetadataSeekable(reader.metadatas, reader.duration, reader.cues),
body = this.result.slice(reader.metadataSize),
newBlob = new Blob([refinedMetadataBuf, body], {
type: "video/webm"
});
callback(newBlob)
}, fileReader.readAsArrayBuffer(inputBlob)
}
function isMediaRecorderCompatible() {
if (isFirefox || isSafari || isEdge) return !0;
var verOffset, ix, nAgt = (navigator.appVersion, navigator.userAgent),
fullVersion = "" + parseFloat(navigator.appVersion),
majorVersion = parseInt(navigator.appVersion, 10);
return (isChrome || isOpera) && (verOffset = nAgt.indexOf("Chrome"), fullVersion = nAgt.substring(verOffset + 7)), (ix = fullVersion.indexOf(";")) !== -1 && (fullVersion = fullVersion.substring(0, ix)), (ix = fullVersion.indexOf(" ")) !== -
1 && (fullVersion = fullVersion.substring(0, ix)), majorVersion = parseInt("" + fullVersion, 10), isNaN(majorVersion) && (fullVersion = "" + parseFloat(navigator.appVersion), majorVersion = parseInt(navigator.appVersion, 10)),
majorVersion >= 49
}
function MediaStreamRecorder(mediaStream, config) {
function updateTimeStamp() {
self.timestamps.push((new Date).getTime()), "function" == typeof config.onTimeStamp && config.onTimeStamp(self.timestamps[self.timestamps.length - 1], self.timestamps)
}
function getMimeType(secondObject) {
return mediaRecorder && mediaRecorder.mimeType ? mediaRecorder.mimeType : secondObject.mimeType || "video/webm"
}
function clearRecordedDataCB() {
arrayOfBlobs = [], mediaRecorder = null, self.timestamps = []
}
function isMediaStreamActive() {
if ("active" in mediaStream) {
if (!mediaStream.active) return !1
} else if ("ended" in mediaStream && mediaStream.ended) return !1;
return !0
}
var self = this;
if ("undefined" == typeof mediaStream) throw 'First argument "MediaStream" is required.';
if ("undefined" == typeof MediaRecorder) throw "Your browser does not support the Media Recorder API. Please try other modules e.g. WhammyRecorder or StereoAudioRecorder.";
if (config = config || {
mimeType: "video/webm"
}, "audio" === config.type) {
if (getTracks(mediaStream, "video").length && getTracks(mediaStream, "audio").length) {
var stream;
navigator.mozGetUserMedia ? (stream = new MediaStream, stream.addTrack(getTracks(mediaStream, "audio")[0])) : stream = new MediaStream(getTracks(mediaStream, "audio")), mediaStream = stream
}
config.mimeType && config.mimeType.toString().toLowerCase().indexOf("audio") !== -1 || (config.mimeType = isChrome ? "audio/webm" : "audio/ogg"), config.mimeType && "audio/ogg" !== config.mimeType.toString().toLowerCase() && navigator
.mozGetUserMedia && (config.mimeType = "audio/ogg")
}
var arrayOfBlobs = [];
this.getArrayOfBlobs = function() {
return arrayOfBlobs
}, this.record = function() {
self.blob = null, self.clearRecordedData(), self.timestamps = [], allStates = [], arrayOfBlobs = [];
var recorderHints = config;
config.disableLogs || console.log("Passing following config over MediaRecorder API.", recorderHints), mediaRecorder && (mediaRecorder = null), isChrome && !isMediaRecorderCompatible() && (recorderHints = "video/vp8"), "function" ==
typeof MediaRecorder.isTypeSupported && recorderHints.mimeType && (MediaRecorder.isTypeSupported(recorderHints.mimeType) || (config.disableLogs || console.warn("MediaRecorder API seems unable to record mimeType:", recorderHints
.mimeType), recorderHints.mimeType = "audio" === config.type ? "audio/webm" : "video/webm"));
try {
mediaRecorder = new MediaRecorder(mediaStream, recorderHints), config.mimeType = recorderHints.mimeType
} catch (e) {
mediaRecorder = new MediaRecorder(mediaStream)
}
recorderHints.mimeType && !MediaRecorder.isTypeSupported && "canRecordMimeType" in mediaRecorder && mediaRecorder.canRecordMimeType(recorderHints.mimeType) === !1 && (config.disableLogs || console.warn(
"MediaRecorder API seems unable to record mimeType:", recorderHints.mimeType)), mediaRecorder.ondataavailable = function(e) {
if (e.data && allStates.push("ondataavailable: " + bytesToSize(e.data.size)), "number" != typeof config.timeSlice) {
if (!e.data || !e.data.size || e.data.size < 100 || self.blob) return void(self.recordingCallback && (self.recordingCallback(new Blob([], {
type: getMimeType(recorderHints)
})), self.recordingCallback = null));
self.blob = config.getNativeBlob ? e.data : new Blob([e.data], {
type: getMimeType(recorderHints)
}), self.recordingCallback && (self.recordingCallback(self.blob), self.recordingCallback = null)
} else if (e.data && e.data.size && (arrayOfBlobs.push(e.data), updateTimeStamp(), "function" == typeof config.ondataavailable)) {
var blob = config.getNativeBlob ? e.data : new Blob([e.data], {
type: getMimeType(recorderHints)
});
config.ondataavailable(blob)
}
}, mediaRecorder.onstart = function() {
allStates.push("started")
}, mediaRecorder.onpause = function() {
allStates.push("paused")
}, mediaRecorder.onresume = function() {
allStates.push("resumed")
}, mediaRecorder.onstop = function() {
allStates.push("stopped")
}, mediaRecorder.onerror = function(error) {
error && (error.name || (error.name = "UnknownError"), allStates.push("error: " + error), config.disableLogs || (error.name.toString().toLowerCase().indexOf("invalidstate") !== -1 ? console.error(
"The MediaRecorder is not in a state in which the proposed operation is allowed to be executed.", error) : error.name.toString().toLowerCase().indexOf("notsupported") !== -1 ? console.error("MIME type (", recorderHints
.mimeType, ") is not supported.", error) : error.name.toString().toLowerCase().indexOf("security") !== -1 ? console.error("MediaRecorder security error", error) : "OutOfMemory" === error.name ? console.error(
"The UA has exhaused the available memory. User agents SHOULD provide as much additional information as possible in the message attribute.", error) : "IllegalStreamModification" === error.name ? console.error(
"A modification to the stream has occurred that makes it impossible to continue recording. An example would be the addition of a Track while recording is occurring. User agents SHOULD provide as much additional information as possible in the message attribute.",
error) : "OtherRecordingError" === error.name ? console.error("Used for an fatal error other than those listed above. User agents SHOULD provide as much additional information as possible in the message attribute.", error) :
"GenericError" === error.name ? console.error("The UA cannot provide the codec or recording option that has been requested.", error) : console.error("MediaRecorder Error", error)), function(looper) {
return !self.manuallyStopped && mediaRecorder && "inactive" === mediaRecorder.state ? (delete config.timeslice, void mediaRecorder.start(6e5)) : void setTimeout(looper, 1e3)
}(), "inactive" !== mediaRecorder.state && "stopped" !== mediaRecorder.state && mediaRecorder.stop())
}, "number" == typeof config.timeSlice ? (updateTimeStamp(), mediaRecorder.start(config.timeSlice)) : mediaRecorder.start(36e5), config.initCallback && config.initCallback()
}, this.timestamps = [], this.stop = function(callback) {
callback = callback || function() {}, self.manuallyStopped = !0, mediaRecorder && (this.recordingCallback = callback, "recording" === mediaRecorder.state && mediaRecorder.stop(), "number" == typeof config.timeSlice && setTimeout(
function() {
self.blob = new Blob(arrayOfBlobs, {
type: getMimeType(config)
}), self.recordingCallback(self.blob)
}, 100))
}, this.pause = function() {
mediaRecorder && "recording" === mediaRecorder.state && mediaRecorder.pause()
}, this.resume = function() {
mediaRecorder && "paused" === mediaRecorder.state && mediaRecorder.resume()
}, this.clearRecordedData = function() {
mediaRecorder && "recording" === mediaRecorder.state && self.stop(clearRecordedDataCB), clearRecordedDataCB()
};
var mediaRecorder;
this.getInternalRecorder = function() {
return mediaRecorder
}, this.blob = null, this.getState = function() {
return mediaRecorder ? mediaRecorder.state || "inactive" : "inactive"
};
var allStates = [];
this.getAllStates = function() {
return allStates
}, "undefined" == typeof config.checkForInactiveTracks && (config.checkForInactiveTracks = !1);
var self = this;
! function looper() {
if (mediaRecorder && config.checkForInactiveTracks !== !1) return isMediaStreamActive() === !1 ? (config.disableLogs || console.log("MediaStream seems stopped."), void self.stop()) : void setTimeout(looper, 1e3)
}(), this.name = "MediaStreamRecorder", this.toString = function() {
return this.name
}
}
function StereoAudioRecorder(mediaStream, config) {
function isMediaStreamActive() {
if (config.checkForInactiveTracks === !1) return !0;
if ("active" in mediaStream) {
if (!mediaStream.active) return !1
} else if ("ended" in mediaStream && mediaStream.ended) return !1;
return !0
}
function mergeLeftRightBuffers(config, callback) {
function mergeAudioBuffers(config, cb) {
function interpolateArray(data, newSampleRate, oldSampleRate) {
var fitCount = Math.round(data.length * (newSampleRate / oldSampleRate)),
newData = [],
springFactor = Number((data.length - 1) / (fitCount - 1));
newData[0] = data[0];
for (var i = 1; i < fitCount - 1; i++) {
var tmp = i * springFactor,
before = Number(Math.floor(tmp)).toFixed(),
after = Number(Math.ceil(tmp)).toFixed(),
atPoint = tmp - before;
newData[i] = linearInterpolate(data[before], data[after], atPoint)
}
return newData[fitCount - 1] = data[data.length - 1], newData
}
function linearInterpolate(before, after, atPoint) {
return before + (after - before) * atPoint
}
function mergeBuffers(channelBuffer, rLength) {
for (var result = new Float64Array(rLength), offset = 0, lng = channelBuffer.length, i = 0; i < lng; i++) {
var buffer = channelBuffer[i];
result.set(buffer, offset), offset += buffer.length
}
return result
}
function interleave(leftChannel, rightChannel) {
for (var length = leftChannel.length + rightChannel.length, result = new Float64Array(length), inputIndex = 0, index = 0; index < length;) result[index++] = leftChannel[inputIndex], result[index++] = rightChannel[inputIndex],
inputIndex++;
return result
}
function writeUTFBytes(view, offset, string) {
for (var lng = string.length, i = 0; i < lng; i++) view.setUint8(offset + i, string.charCodeAt(i))
}
var numberOfAudioChannels = config.numberOfAudioChannels,
leftBuffers = config.leftBuffers.slice(0),
rightBuffers = config.rightBuffers.slice(0),
sampleRate = config.sampleRate,
internalInterleavedLength = config.internalInterleavedLength,
desiredSampRate = config.desiredSampRate;
2 === numberOfAudioChannels && (leftBuffers = mergeBuffers(leftBuffers, internalInterleavedLength), rightBuffers = mergeBuffers(rightBuffers, internalInterleavedLength), desiredSampRate && (leftBuffers = interpolateArray(leftBuffers,
desiredSampRate, sampleRate), rightBuffers = interpolateArray(rightBuffers, desiredSampRate, sampleRate))), 1 === numberOfAudioChannels && (leftBuffers = mergeBuffers(leftBuffers, internalInterleavedLength), desiredSampRate && (
leftBuffers = interpolateArray(leftBuffers, desiredSampRate, sampleRate))), desiredSampRate && (sampleRate = desiredSampRate);
var interleaved;
2 === numberOfAudioChannels && (interleaved = interleave(leftBuffers, rightBuffers)), 1 === numberOfAudioChannels && (interleaved = leftBuffers);
var interleavedLength = interleaved.length,
resultingBufferLength = 44 + 2 * interleavedLength,
buffer = new ArrayBuffer(resultingBufferLength),
view = new DataView(buffer);
writeUTFBytes(view, 0, "RIFF"), view.setUint32(4, 36 + 2 * interleavedLength, !0), writeUTFBytes(view, 8, "WAVE"), writeUTFBytes(view, 12, "fmt "), view.setUint32(16, 16, !0), view.setUint16(20, 1, !0), view.setUint16(22,
numberOfAudioChannels, !0), view.setUint32(24, sampleRate, !0), view.setUint32(28, sampleRate * numberOfAudioChannels * 2, !0), view.setUint16(32, 2 * numberOfAudioChannels, !0), view.setUint16(34, 16, !0), writeUTFBytes(view, 36,
"data"), view.setUint32(40, 2 * interleavedLength, !0);
for (var lng = interleavedLength, index = 44, volume = 1, i = 0; i < lng; i++) view.setInt16(index, interleaved[i] * (32767 * volume), !0), index += 2;
return cb ? cb({
buffer: buffer,
view: view
}) : void postMessage({
buffer: buffer,
view: view
})
}
if (config.noWorker) return void mergeAudioBuffers(config, function(data) {
callback(data.buffer, data.view)
});
var webWorker = processInWebWorker(mergeAudioBuffers);
webWorker.onmessage = function(event) {
callback(event.data.buffer, event.data.view), URL.revokeObjectURL(webWorker.workerURL), webWorker.terminate()
}, webWorker.postMessage(config)
}
function processInWebWorker(_function) {
var workerURL = URL.createObjectURL(new Blob([_function.toString(), ";this.onmessage = function (eee) {" + _function.name + "(eee.data);}"], {
type: "application/javascript"
})),
worker = new Worker(workerURL);
return worker.workerURL = workerURL, worker
}
function resetVariables() {
leftchannel = [], rightchannel = [], recordingLength = 0, isAudioProcessStarted = !1, recording = !1, isPaused = !1, context = null, self.leftchannel = leftchannel, self.rightchannel = rightchannel, self.numberOfAudioChannels =
numberOfAudioChannels, self.desiredSampRate = desiredSampRate, self.sampleRate = sampleRate, self.recordingLength = recordingLength, intervalsBasedBuffers = {
left: [],
right: [],
recordingLength: 0
}
}
function clearRecordedDataCB() {
jsAudioNode && (jsAudioNode.onaudioprocess = null, jsAudioNode.disconnect(), jsAudioNode = null), audioInput && (audioInput.disconnect(), audioInput = null), resetVariables()
}
function onAudioProcessDataAvailable(e) {
if (!isPaused) {
if (isMediaStreamActive() === !1 && (config.disableLogs || console.log("MediaStream seems stopped."), jsAudioNode.disconnect(), recording = !1), !recording) return void(audioInput && (audioInput.disconnect(), audioInput = null));
isAudioProcessStarted || (isAudioProcessStarted = !0, config.onAudioProcessStarted && config.onAudioProcessStarted(), config.initCallback && config.initCallback());
var left = e.inputBuffer.getChannelData(0),
chLeft = new Float32Array(left);
if (leftchannel.push(chLeft), 2 === numberOfAudioChannels) {
var right = e.inputBuffer.getChannelData(1),
chRight = new Float32Array(right);
rightchannel.push(chRight)
}
recordingLength += bufferSize, self.recordingLength = recordingLength, "undefined" != typeof config.timeSlice && (intervalsBasedBuffers.recordingLength += bufferSize, intervalsBasedBuffers.left.push(chLeft), 2 ===
numberOfAudioChannels && intervalsBasedBuffers.right.push(chRight))
}
}
function looper() {
recording && "function" == typeof config.ondataavailable && "undefined" != typeof config.timeSlice && (intervalsBasedBuffers.left.length ? (mergeLeftRightBuffers({
desiredSampRate: desiredSampRate,
sampleRate: sampleRate,
numberOfAudioChannels: numberOfAudioChannels,
internalInterleavedLength: intervalsBasedBuffers.recordingLength,
leftBuffers: intervalsBasedBuffers.left,
rightBuffers: 1 === numberOfAudioChannels ? [] : intervalsBasedBuffers.right
}, function(buffer, view) {
var blob = new Blob([view], {
type: "audio/wav"
});
config.ondataavailable(blob), setTimeout(looper, config.timeSlice)
}), intervalsBasedBuffers = {
left: [],
right: [],
recordingLength: 0
}) : setTimeout(looper, config.timeSlice))
}
if (!getTracks(mediaStream, "audio").length) throw "Your stream has no audio tracks.";
config = config || {};
var jsAudioNode, self = this,
leftchannel = [],
rightchannel = [],
recording = !1,
recordingLength = 0,
numberOfAudioChannels = 2,
desiredSampRate = config.desiredSampRate;
config.leftChannel === !0 && (numberOfAudioChannels = 1), 1 === config.numberOfAudioChannels && (numberOfAudioChannels = 1), (!numberOfAudioChannels || numberOfAudioChannels < 1) && (numberOfAudioChannels = 2), config.disableLogs || console
.log("StereoAudioRecorder is set to record number of channels: " + numberOfAudioChannels), "undefined" == typeof config.checkForInactiveTracks && (config.checkForInactiveTracks = !0), this.record = function() {
if (isMediaStreamActive() === !1) throw "Please make sure MediaStream is active.";
resetVariables(), isAudioProcessStarted = isPaused = !1, recording = !0, "undefined" != typeof config.timeSlice && looper()
}, this.stop = function(callback) {
callback = callback || function() {}, recording = !1, mergeLeftRightBuffers({
desiredSampRate: desiredSampRate,
sampleRate: sampleRate,
numberOfAudioChannels: numberOfAudioChannels,
internalInterleavedLength: recordingLength,
leftBuffers: leftchannel,
rightBuffers: 1 === numberOfAudioChannels ? [] : rightchannel,
noWorker: config.noWorker
}, function(buffer, view) {
self.blob = new Blob([view], {
type: "audio/wav"
}), self.buffer = new ArrayBuffer(view.buffer.byteLength), self.view = view, self.sampleRate = desiredSampRate || sampleRate, self.bufferSize = bufferSize, self.length = recordingLength, isAudioProcessStarted = !1, callback &&
callback(self.blob)
})
}, "undefined" == typeof RecordRTC.Storage && (RecordRTC.Storage = {
AudioContextConstructor: null,
AudioContext: window.AudioContext || window.webkitAudioContext
}), RecordRTC.Storage.AudioContextConstructor && "closed" !== RecordRTC.Storage.AudioContextConstructor.state || (RecordRTC.Storage.AudioContextConstructor = new RecordRTC.Storage.AudioContext);
var context = RecordRTC.Storage.AudioContextConstructor,
audioInput = context.createMediaStreamSource(mediaStream),
legalBufferValues = [0, 256, 512, 1024, 2048, 4096, 8192, 16384],
bufferSize = "undefined" == typeof config.bufferSize ? 4096 : config.bufferSize;
if (legalBufferValues.indexOf(bufferSize) === -1 && (config.disableLogs || console.log("Legal values for buffer-size are " + JSON.stringify(legalBufferValues, null, "\t"))), context.createJavaScriptNode) jsAudioNode = context
.createJavaScriptNode(bufferSize, numberOfAudioChannels, numberOfAudioChannels);
else {
if (!context.createScriptProcessor) throw "WebAudio API has no support on this browser.";
jsAudioNode = context.createScriptProcessor(bufferSize, numberOfAudioChannels, numberOfAudioChannels)
}
audioInput.connect(jsAudioNode), config.bufferSize || (bufferSize = jsAudioNode.bufferSize);
var sampleRate = "undefined" != typeof config.sampleRate ? config.sampleRate : context.sampleRate || 44100;
(sampleRate < 22050 || sampleRate > 96e3) && (config.disableLogs || console.log("sample-rate must be under range 22050 and 96000.")), config.disableLogs || config.desiredSampRate && console.log("Desired sample-rate: " + config
.desiredSampRate);
var isPaused = !1;
this.pause = function() {
isPaused = !0
}, this.resume = function() {
if (isMediaStreamActive() === !1) throw "Please make sure MediaStream is active.";
return recording ? void(isPaused = !1) : (config.disableLogs || console.log("Seems recording has been restarted."), void this.record())
}, this.clearRecordedData = function() {
config.checkForInactiveTracks = !1, recording && this.stop(clearRecordedDataCB), clearRecordedDataCB()
}, this.name = "StereoAudioRecorder", this.toString = function() {
return this.name
};
var isAudioProcessStarted = !1;
jsAudioNode.onaudioprocess = onAudioProcessDataAvailable, context.createMediaStreamDestination ? jsAudioNode.connect(context.createMediaStreamDestination()) : jsAudioNode.connect(context.destination), this.leftchannel = leftchannel, this
.rightchannel = rightchannel, this.numberOfAudioChannels = numberOfAudioChannels, this.desiredSampRate = desiredSampRate, this.sampleRate = sampleRate, self.recordingLength = recordingLength;
var intervalsBasedBuffers = {
left: [],
right: [],
recordingLength: 0
}
}
function CanvasRecorder(htmlElement, config) {
function clearRecordedDataCB() {
whammy.frames = [], isRecording = !1, isPausedRecording = !1
}
function cloneCanvas() {
var newCanvas = document.createElement("canvas"),
context = newCanvas.getContext("2d");
return newCanvas.width = htmlElement.width, newCanvas.height = htmlElement.height, context.drawImage(htmlElement, 0, 0), newCanvas
}
function drawCanvasFrame() {
if (isPausedRecording) return lastTime = (new Date).getTime(), setTimeout(drawCanvasFrame, 500);
if ("canvas" === htmlElement.nodeName.toLowerCase()) {
var duration = (new Date).getTime() - lastTime;
return lastTime = (new Date).getTime(), whammy.frames.push({
image: cloneCanvas(),
duration: duration
}), void(isRecording && setTimeout(drawCanvasFrame, config.frameInterval))
}
html2canvas(htmlElement, {
grabMouse: "undefined" == typeof config.showMousePointer || config.showMousePointer,
onrendered: function(canvas) {
var duration = (new Date).getTime() - lastTime;
return duration ? (lastTime = (new Date).getTime(), whammy.frames.push({
image: canvas.toDataURL("image/webp", 1),
duration: duration
}), void(isRecording && setTimeout(drawCanvasFrame, config.frameInterval))) : setTimeout(drawCanvasFrame, config.frameInterval)
}
})
}
if ("undefined" == typeof html2canvas) throw "Please link: https://www.webrtc-experiment.com/screenshot.js";
config = config || {}, config.frameInterval || (config.frameInterval = 10);
var isCanvasSupportsStreamCapturing = !1;
["captureStream", "mozCaptureStream", "webkitCaptureStream"].forEach(function(item) {
item in document.createElement("canvas") && (isCanvasSupportsStreamCapturing = !0)
});
var _isChrome = !(!window.webkitRTCPeerConnection && !window.webkitGetUserMedia || !window.chrome),
chromeVersion = 50,
matchArray = navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./);
_isChrome && matchArray && matchArray[2] && (chromeVersion = parseInt(matchArray[2], 10)), _isChrome && chromeVersion < 52 && (isCanvasSupportsStreamCapturing = !1), config.useWhammyRecorder && (isCanvasSupportsStreamCapturing = !1);
var globalCanvas, mediaStreamRecorder;
if (isCanvasSupportsStreamCapturing)
if (config.disableLogs || console.log("Your browser supports both MediRecorder API and canvas.captureStream!"), htmlElement instanceof HTMLCanvasElement) globalCanvas = htmlElement;
else {
if (!(htmlElement instanceof CanvasRenderingContext2D)) throw "Please pass either HTMLCanvasElement or CanvasRenderingContext2D.";
globalCanvas = htmlElement.canvas
}
else navigator.mozGetUserMedia && (config.disableLogs || console.error("Canvas recording is NOT supported in Firefox."));
var isRecording;
this.record = function() {
if (isRecording = !0, isCanvasSupportsStreamCapturing && !config.useWhammyRecorder) {
var canvasMediaStream;
"captureStream" in globalCanvas ? canvasMediaStream = globalCanvas.captureStream(25) : "mozCaptureStream" in globalCanvas ? canvasMediaStream = globalCanvas.mozCaptureStream(25) : "webkitCaptureStream" in globalCanvas && (
canvasMediaStream = globalCanvas.webkitCaptureStream(25));
try {
var mdStream = new MediaStream;
mdStream.addTrack(getTracks(canvasMediaStream, "video")[0]), canvasMediaStream = mdStream
} catch (e) {}
if (!canvasMediaStream) throw "captureStream API are NOT available.";
mediaStreamRecorder = new MediaStreamRecorder(canvasMediaStream, {
mimeType: config.mimeType || "video/webm"
}), mediaStreamRecorder.record()
} else whammy.frames = [], lastTime = (new Date).getTime(), drawCanvasFrame();
config.initCallback && config.initCallback()
}, this.getWebPImages = function(callback) {
if ("canvas" !== htmlElement.nodeName.toLowerCase()) return void callback();
var framesLength = whammy.frames.length;
whammy.frames.forEach(function(frame, idx) {
var framesRemaining = framesLength - idx;
config.disableLogs || console.log(framesRemaining + "/" + framesLength + " frames remaining"), config.onEncodingCallback && config.onEncodingCallback(framesRemaining, framesLength);
var webp = frame.image.toDataURL("image/webp", 1);
whammy.frames[idx].image = webp
}), config.disableLogs || console.log("Generating WebM"), callback()
}, this.stop = function(callback) {
isRecording = !1;
var that = this;
return isCanvasSupportsStreamCapturing && mediaStreamRecorder ? void mediaStreamRecorder.stop(callback) : void this.getWebPImages(function() {
whammy.compile(function(blob) {
config.disableLogs || console.log("Recording finished!"), that.blob = blob, that.blob.forEach && (that.blob = new Blob([], {
type: "video/webm"
})), callback && callback(that.blob), whammy.frames = []
})
})
};
var isPausedRecording = !1;
this.pause = function() {
if (isPausedRecording = !0, mediaStreamRecorder instanceof MediaStreamRecorder) return void mediaStreamRecorder.pause()
}, this.resume = function() {
return isPausedRecording = !1, mediaStreamRecorder instanceof MediaStreamRecorder ? void mediaStreamRecorder.resume() : void(isRecording || this.record())
}, this.clearRecordedData = function() {
isRecording && this.stop(clearRecordedDataCB), clearRecordedDataCB()
}, this.name = "CanvasRecorder", this.toString = function() {
return this.name
};
var lastTime = (new Date).getTime(),
whammy = new Whammy.Video(100)
}
function WhammyRecorder(mediaStream, config) {
function drawFrames(frameInterval) {
frameInterval = "undefined" != typeof frameInterval ? frameInterval : 10;
var duration = (new Date).getTime() - lastTime;
return duration ? isPausedRecording ? (lastTime = (new Date).getTime(), setTimeout(drawFrames, 100)) : (lastTime = (new Date).getTime(), video.paused && video.play(), context.drawImage(video, 0, 0, canvas.width, canvas.height), whammy
.frames.push({
duration: duration,
image: canvas.toDataURL("image/webp")
}), void(isStopDrawing || setTimeout(drawFrames, frameInterval, frameInterval))) : setTimeout(drawFrames, frameInterval, frameInterval)
}
function asyncLoop(o) {
var i = -1,
length = o.length;
! function loop() {
return i++, i === length ? void o.callback() : void setTimeout(function() {
o.functionToLoop(loop, i)
}, 1)
}()
}
function dropBlackFrames(_frames, _framesToCheck, _pixTolerance, _frameTolerance, callback) {
var localCanvas = document.createElement("canvas");
localCanvas.width = canvas.width, localCanvas.height = canvas.height;
var context2d = localCanvas.getContext("2d"),
resultFrames = [],
checkUntilNotBlack = _framesToCheck === -1,
endCheckFrame = _framesToCheck && _framesToCheck > 0 && _framesToCheck <= _frames.length ? _framesToCheck : _frames.length,
sampleColor = {
r: 0,
g: 0,
b: 0
},
maxColorDifference = Math.sqrt(Math.pow(255, 2) + Math.pow(255, 2) + Math.pow(255, 2)),
pixTolerance = _pixTolerance && _pixTolerance >= 0 && _pixTolerance <= 1 ? _pixTolerance : 0,
frameTolerance = _frameTolerance && _frameTolerance >= 0 && _frameTolerance <= 1 ? _frameTolerance : 0,
doNotCheckNext = !1;
asyncLoop({
length: endCheckFrame,
functionToLoop: function(loop, f) {
var matchPixCount, endPixCheck, maxPixCount, finishImage = function() {
!doNotCheckNext && maxPixCount - matchPixCount <= maxPixCount * frameTolerance || (checkUntilNotBlack && (doNotCheckNext = !0), resultFrames.push(_frames[f])), loop()
};
if (doNotCheckNext) finishImage();
else {
var image = new Image;
image.onload = function() {
context2d.drawImage(image, 0, 0, canvas.width, canvas.height);
var imageData = context2d.getImageData(0, 0, canvas.width, canvas.height);
matchPixCount = 0, endPixCheck = imageData.data.length, maxPixCount = imageData.data.length / 4;
for (var pix = 0; pix < endPixCheck; pix += 4) {
var currentColor = {
r: imageData.data[pix],
g: imageData.data[pix + 1],
b: imageData.data[pix + 2]
},
colorDifference = Math.sqrt(Math.pow(currentColor.r - sampleColor.r, 2) + Math.pow(currentColor.g - sampleColor.g, 2) + Math.pow(currentColor.b - sampleColor.b, 2));
colorDifference <= maxColorDifference * pixTolerance && matchPixCount++
}
finishImage()
}, image.src = _frames[f].image
}
},
callback: function() {
resultFrames = resultFrames.concat(_frames.slice(endCheckFrame)), resultFrames.length <= 0 && resultFrames.push(_frames[_frames.length - 1]), callback(resultFrames)
}
})
}
function clearRecordedDataCB() {
whammy.frames = [], isStopDrawing = !0, isPausedRecording = !1
}
config = config || {}, config.frameInterval || (config.frameInterval = 10), config.disableLogs || console.log("Using frames-interval:", config.frameInterval), this.record = function() {
config.width || (config.width = 320), config.height || (config.height = 240), config.video || (config.video = {
width: config.width,
height: config.height
}), config.canvas || (config.canvas = {
width: config.width,
height: config.height
}), canvas.width = config.canvas.width || 320, canvas.height = config.canvas.height || 240, context = canvas.getContext("2d"), config.video && config.video instanceof HTMLVideoElement ? (video = config.video.cloneNode(), config
.initCallback && config.initCallback()) : (video = document.createElement("video"), setSrcObject(mediaStream, video), video.onloadedmetadata = function() {
config.initCallback && config.initCallback()
}, video.width = config.video.width, video.height = config.video.height), video.muted = !0, video.play(), lastTime = (new Date).getTime(), whammy = new Whammy.Video, config.disableLogs || (console.log("canvas resolutions", canvas
.width, "*", canvas.height), console.log("video width/height", video.width || canvas.width, "*", video.height || canvas.height)), drawFrames(config.frameInterval)
};
var isStopDrawing = !1;
this.stop = function(callback) {
callback = callback || function() {}, isStopDrawing = !0;
var _this = this;
setTimeout(function() {
dropBlackFrames(whammy.frames, -1, null, null, function(frames) {
whammy.frames = frames, config.advertisement && config.advertisement.length && (whammy.frames = config.advertisement.concat(whammy.frames)), whammy.compile(function(blob) {
_this.blob = blob, _this.blob.forEach && (_this.blob = new Blob([], {
type: "video/webm"
})), callback && callback(_this.blob)
})
})
}, 10)
};
var isPausedRecording = !1;
this.pause = function() {
isPausedRecording = !0
}, this.resume = function() {
isPausedRecording = !1, isStopDrawing && this.record()
}, this.clearRecordedData = function() {
isStopDrawing || this.stop(clearRecordedDataCB), clearRecordedDataCB()
}, this.name = "WhammyRecorder", this.toString = function() {
return this.name
};
var video, lastTime, whammy, canvas = document.createElement("canvas"),
context = canvas.getContext("2d")
}
function GifRecorder(mediaStream, config) {
function clearRecordedDataCB() {
gifEncoder && (gifEncoder.stream().bin = [])
}
if ("undefined" == typeof GIFEncoder) {
var script = document.createElement("script");
script.src = "https://www.webrtc-experiment.com/gif-recorder.js", (document.body || document.documentElement).appendChild(script)
}
config = config || {};
var isHTMLObject = mediaStream instanceof CanvasRenderingContext2D || mediaStream instanceof HTMLCanvasElement;
this.record = function() {
function drawVideoFrame(time) {
if (self.clearedRecordedData !== !0) {
if (isPausedRecording) return setTimeout(function() {
drawVideoFrame(time)
}, 100);
lastAnimationFrame = requestAnimationFrame(drawVideoFrame), void 0 === typeof lastFrameTime && (lastFrameTime = time), time - lastFrameTime < 90 || (!isHTMLObject && video.paused && video.play(), isHTMLObject || context.drawImage(
video, 0, 0, canvas.width, canvas.height), config.onGifPreview && config.onGifPreview(canvas.toDataURL("image/png")), gifEncoder.addFrame(context), lastFrameTime = time)
}
}
return "undefined" == typeof GIFEncoder ? void setTimeout(self.record, 1e3) : isLoadedMetaData ? (isHTMLObject || (config.width || (config.width = video.offsetWidth || 320), config.height || (config.height = video.offsetHeight || 240),
config.video || (config.video = {
width: config.width,
height: config.height
}), config.canvas || (config.canvas = {
width: config.width,
height: config.height
}), canvas.width = config.canvas.width || 320, canvas.height = config.canvas.height || 240, video.width = config.video.width || 320, video.height = config.video.height || 240), gifEncoder = new GIFEncoder, gifEncoder.setRepeat(0),
gifEncoder.setDelay(config.frameRate || 200), gifEncoder.setQuality(config.quality || 10), gifEncoder.start(), "function" == typeof config.onGifRecordingStarted && config.onGifRecordingStarted(), startTime = Date.now(),
lastAnimationFrame = requestAnimationFrame(drawVideoFrame), void(config.initCallback && config.initCallback())) : void setTimeout(self.record, 1e3)
}, this.stop = function(callback) {
callback = callback || function() {}, lastAnimationFrame && cancelAnimationFrame(lastAnimationFrame), endTime = Date.now(), this.blob = new Blob([new Uint8Array(gifEncoder.stream().bin)], {
type: "image/gif"
}), callback(this.blob), gifEncoder.stream().bin = []
};
var isPausedRecording = !1;
this.pause = function() {
isPausedRecording = !0
}, this.resume = function() {
isPausedRecording = !1
}, this.clearRecordedData = function() {
self.clearedRecordedData = !0, clearRecordedDataCB()
}, this.name = "GifRecorder", this.toString = function() {
return this.name
};
var canvas = document.createElement("canvas"),
context = canvas.getContext("2d");
isHTMLObject && (mediaStream instanceof CanvasRenderingContext2D ? (context = mediaStream, canvas = context.canvas) : mediaStream instanceof HTMLCanvasElement && (context = mediaStream.getContext("2d"), canvas = mediaStream));
var isLoadedMetaData = !0;
if (!isHTMLObject) {
var video = document.createElement("video");
video.muted = !0, video.autoplay = !0, video.playsInline = !0, isLoadedMetaData = !1, video.onloadedmetadata = function() {
isLoadedMetaData = !0
}, setSrcObject(mediaStream, video), video.play()
}
var startTime, endTime, lastFrameTime, gifEncoder, lastAnimationFrame = null,
self = this
}
function MultiStreamsMixer(arrayOfMediaStreams, elementClass) {
function setSrcObject(stream, element) {
"srcObject" in element ? element.srcObject = stream : "mozSrcObject" in element ? element.mozSrcObject = stream : element.srcObject = stream
}
function drawVideosToCanvas() {
if (!isStopDrawingFrames) {
var videosLength = videos.length,
fullcanvas = !1,
remaining = [];
if (videos.forEach(function(video) {
video.stream || (video.stream = {}), video.stream.fullcanvas ? fullcanvas = video : remaining.push(video)
}), fullcanvas) canvas.width = fullcanvas.stream.width, canvas.height = fullcanvas.stream.height;
else if (remaining.length) {
canvas.width = videosLength > 1 ? 2 * remaining[0].width : remaining[0].width;
var height = 1;
3 !== videosLength && 4 !== videosLength || (height = 2), 5 !== videosLength && 6 !== videosLength || (height = 3), 7 !== videosLength && 8 !== videosLength || (height = 4), 9 !== videosLength && 10 !== videosLength || (height = 5),
canvas.height = remaining[0].height * height
} else canvas.width = self.width || 360, canvas.height = self.height || 240;
fullcanvas && fullcanvas instanceof HTMLVideoElement && drawImage(fullcanvas), remaining.forEach(function(video, idx) {
drawImage(video, idx)
}), setTimeout(drawVideosToCanvas, self.frameInterval)
}
}
function drawImage(video, idx) {
if (!isStopDrawingFrames) {
var x = 0,
y = 0,
width = video.width,
height = video.height;
1 === idx && (x = video.width), 2 === idx && (y = video.height), 3 === idx && (x = video.width, y = video.height), 4 === idx && (y = 2 * video.height), 5 === idx && (x = video.width, y = 2 * video.height), 6 === idx && (y = 3 * video
.height), 7 === idx && (x = video.width, y = 3 * video.height), "undefined" != typeof video.stream.left && (x = video.stream.left), "undefined" != typeof video.stream.top && (y = video.stream.top), "undefined" != typeof video.stream
.width && (width = video.stream.width), "undefined" != typeof video.stream.height && (height = video.stream.height), context.drawImage(video, x, y, width, height), "function" == typeof video.stream.onRender && video.stream.onRender(
context, x, y, width, height, idx)
}
}
function getMixedStream() {
isStopDrawingFrames = !1;
var mixedVideoStream = getMixedVideoStream(),
mixedAudioStream = getMixedAudioStream();
mixedAudioStream && mixedAudioStream.getTracks().filter(function(t) {
return "audio" === t.kind
}).forEach(function(track) {
mixedVideoStream.addTrack(track)
});
var fullcanvas;
return arrayOfMediaStreams.forEach(function(stream) {
stream.fullcanvas && (fullcanvas = !0)
}), mixedVideoStream
}
function getMixedVideoStream() {
resetVideoStreams();
var capturedStream;
"captureStream" in canvas ? capturedStream = canvas.captureStream() : "mozCaptureStream" in canvas ? capturedStream = canvas.mozCaptureStream() : self.disableLogs || console.error(
"Upgrade to latest Chrome or otherwise enable this flag: chrome://flags/#enable-experimental-web-platform-features");
var videoStream = new MediaStream;
return capturedStream.getTracks().filter(function(t) {
return "video" === t.kind
}).forEach(function(track) {
videoStream.addTrack(track)
}), canvas.stream = videoStream, videoStream
}
function getMixedAudioStream() {
Storage.AudioContextConstructor || (Storage.AudioContextConstructor = new Storage.AudioContext), self.audioContext = Storage.AudioContextConstructor, self.audioSources = [], self.useGainNode === !0 && (self.gainNode = self.audioContext
.createGain(), self.gainNode.connect(self.audioContext.destination), self.gainNode.gain.value = 0);
var audioTracksLength = 0;
if (arrayOfMediaStreams.forEach(function(stream) {
if (stream.getTracks().filter(function(t) {
return "audio" === t.kind
}).length) {
audioTracksLength++;
var audioSource = self.audioContext.createMediaStreamSource(stream);
self.useGainNode === !0 && audioSource.connect(self.gainNode), self.audioSources.push(audioSource)
}
}), audioTracksLength) return self.audioDestination = self.audioContext.createMediaStreamDestination(), self.audioSources.forEach(function(audioSource) {
audioSource.connect(self.audioDestination)
}), self.audioDestination.stream
}
function getVideo(stream) {
var video = document.createElement("video");
return setSrcObject(stream, video), video.className = elementClass, video.muted = !0, video.volume = 0, video.width = stream.width || self.width || 360, video.height = stream.height || self.height || 240, video.play(), video
}
function resetVideoStreams(streams) {
videos = [], streams = streams || arrayOfMediaStreams, streams.forEach(function(stream) {
if (stream.getTracks().filter(function(t) {
return "video" === t.kind
}).length) {
var video = getVideo(stream);
video.stream = stream, videos.push(video)
}
})
}
var browserFakeUserAgent = "Fake/5.0 (FakeOS) AppleWebKit/123 (KHTML, like Gecko) Fake/12.3.4567.89 Fake/123.45";
! function(that) {
"undefined" == typeof RecordRTC && that && "undefined" == typeof window && "undefined" != typeof global && (global.navigator = {
userAgent: browserFakeUserAgent,
getUserMedia: function() {}
}, global.console || (global.console = {}), "undefined" != typeof global.console.log && "undefined" != typeof global.console.error || (global.console.error = global.console.log = global.console.log || function() {
console.log(arguments)
}), "undefined" == typeof document && (that.document = {
documentElement: {
appendChild: function() {
return ""
}
}
}, document.createElement = document.captureStream = document.mozCaptureStream = function() {
var obj = {
getContext: function() {
return obj
},
play: function() {},
pause: function() {},
drawImage: function() {},
toDataURL: function() {
return ""
},
style: {}
};
return obj
}, that.HTMLVideoElement = function() {}), "undefined" == typeof location && (that.location = {
protocol: "file:",
href: "",
hash: ""
}), "undefined" == typeof screen && (that.screen = {
width: 0,
height: 0
}), "undefined" == typeof URL && (that.URL = {
createObjectURL: function() {
return ""
},
revokeObjectURL: function() {
return ""
}
}), that.window = global)
}("undefined" != typeof global ? global : null), elementClass = elementClass || "multi-streams-mixer";
var videos = [],
isStopDrawingFrames = !1,
canvas = document.createElement("canvas"),
context = canvas.getContext("2d");
canvas.style.opacity = 0, canvas.style.position = "absolute", canvas.style.zIndex = -1, canvas.style.top = "-1000em", canvas.style.left = "-1000em", canvas.className = elementClass, (document.body || document.documentElement).appendChild(
canvas), this.disableLogs = !1, this.frameInterval = 10, this.width = 360, this.height = 240, this.useGainNode = !0;
var self = this,
AudioContext = window.AudioContext;
"undefined" == typeof AudioContext && ("undefined" != typeof webkitAudioContext && (AudioContext = webkitAudioContext), "undefined" != typeof mozAudioContext && (AudioContext = mozAudioContext));
var URL = window.URL;
"undefined" == typeof URL && "undefined" != typeof webkitURL && (URL = webkitURL), "undefined" != typeof navigator && "undefined" == typeof navigator.getUserMedia && ("undefined" != typeof navigator.webkitGetUserMedia && (navigator
.getUserMedia = navigator.webkitGetUserMedia), "undefined" != typeof navigator.mozGetUserMedia && (navigator.getUserMedia = navigator.mozGetUserMedia));
var MediaStream = window.MediaStream;
"undefined" == typeof MediaStream && "undefined" != typeof webkitMediaStream && (MediaStream = webkitMediaStream), "undefined" != typeof MediaStream && "undefined" == typeof MediaStream.prototype.stop && (MediaStream.prototype.stop =
function() {
this.getTracks().forEach(function(track) {
track.stop()
})
});
var Storage = {};
"undefined" != typeof AudioContext ? Storage.AudioContext = AudioContext : "undefined" != typeof webkitAudioContext && (Storage.AudioContext = webkitAudioContext), this.startDrawingFrames = function() {
drawVideosToCanvas()
}, this.appendStreams = function(streams) {
if (!streams) throw "First parameter is required.";
streams instanceof Array || (streams = [streams]), streams.forEach(function(stream) {
var newStream = new MediaStream;
if (stream.getTracks().filter(function(t) {
return "video" === t.kind
}).length) {
var video = getVideo(stream);
video.stream = stream, videos.push(video), newStream.addTrack(stream.getTracks().filter(function(t) {
return "video" === t.kind
})[0])
}
if (stream.getTracks().filter(function(t) {
return "audio" === t.kind
}).length) {
var audioSource = self.audioContext.createMediaStreamSource(stream);
self.audioDestination = self.audioContext.createMediaStreamDestination(), audioSource.connect(self.audioDestination), newStream.addTrack(self.audioDestination.stream.getTracks().filter(function(t) {
return "audio" === t.kind
})[0])
}
arrayOfMediaStreams.push(newStream)
})
}, this.releaseStreams = function() {
videos = [], isStopDrawingFrames = !0, self.gainNode && (self.gainNode.disconnect(), self.gainNode = null), self.audioSources.length && (self.audioSources.forEach(function(source) {
source.disconnect()
}), self.audioSources = []), self.audioDestination && (self.audioDestination.disconnect(), self.audioDestination = null), self.audioContext && self.audioContext.close(), self.audioContext = null, context.clearRect(0, 0, canvas.width,
canvas.height), canvas.stream && (canvas.stream.stop(), canvas.stream = null)
}, this.resetVideoStreams = function(streams) {
!streams || streams instanceof Array || (streams = [streams]), resetVideoStreams(streams)
}, this.name = "MultiStreamsMixer", this.toString = function() {
return this.name
}, this.getMixedStream = getMixedStream
}
function MultiStreamRecorder(arrayOfMediaStreams, options) {
function getAllVideoTracks() {
var tracks = [];
return arrayOfMediaStreams.forEach(function(stream) {
getTracks(stream, "video").forEach(function(track) {
tracks.push(track)
})
}), tracks
}
arrayOfMediaStreams = arrayOfMediaStreams || [];
var mixer, mediaRecorder, self = this;
options = options || {
elementClass: "multi-streams-mixer",
mimeType: "video/webm",
video: {
width: 360,
height: 240
}
}, options.frameInterval || (options.frameInterval = 10), options.video || (options.video = {}), options.video.width || (options.video.width = 360), options.video.height || (options.video.height = 240), this.record = function() {
mixer = new MultiStreamsMixer(arrayOfMediaStreams, options.elementClass || "multi-streams-mixer"), getAllVideoTracks().length && (mixer.frameInterval = options.frameInterval || 10, mixer.width = options.video.width || 360, mixer
.height = options.video.height || 240, mixer.startDrawingFrames()), options.previewStream && "function" == typeof options.previewStream && options.previewStream(mixer.getMixedStream()), mediaRecorder = new MediaStreamRecorder(mixer
.getMixedStream(), options), mediaRecorder.record()
}, this.stop = function(callback) {
mediaRecorder && mediaRecorder.stop(function(blob) {
self.blob = blob, callback(blob), self.clearRecordedData()
})
}, this.pause = function() {
mediaRecorder && mediaRecorder.pause()
}, this.resume = function() {
mediaRecorder && mediaRecorder.resume()
}, this.clearRecordedData = function() {
mediaRecorder && (mediaRecorder.clearRecordedData(), mediaRecorder = null), mixer && (mixer.releaseStreams(), mixer = null)
}, this.addStreams = function(streams) {
if (!streams) throw "First parameter is required.";
streams instanceof Array || (streams = [streams]), arrayOfMediaStreams.concat(streams), mediaRecorder && mixer && (mixer.appendStreams(streams), options.previewStream && "function" == typeof options.previewStream && options
.previewStream(mixer.getMixedStream()))
}, this.resetVideoStreams = function(streams) {
mixer && (!streams || streams instanceof Array || (streams = [streams]), mixer.resetVideoStreams(streams))
}, this.getMixer = function() {
return mixer
}, this.name = "MultiStreamRecorder", this.toString = function() {
return this.name
}
}
function RecordRTCPromisesHandler(mediaStream, options) {
if (!this) throw 'Use "new RecordRTCPromisesHandler()"';
if ("undefined" == typeof mediaStream) throw 'First argument "MediaStream" is required.';
var self = this;
self.recordRTC = new RecordRTC(mediaStream, options), this.startRecording = function() {
return new Promise(function(resolve, reject) {
try {
self.recordRTC.startRecording(), resolve()
} catch (e) {
reject(e)
}
})
}, this.stopRecording = function() {
return new Promise(function(resolve, reject) {
try {
self.recordRTC.stopRecording(function(url) {
return self.blob = self.recordRTC.getBlob(), self.blob && self.blob.size ? void resolve(url) : void reject("Empty blob.", self.blob)
})
} catch (e) {
reject(e)
}
})
}, this.pauseRecording = function() {
return new Promise(function(resolve, reject) {
try {
self.recordRTC.pauseRecording(), resolve()
} catch (e) {
reject(e)
}
})
}, this.resumeRecording = function() {
return new Promise(function(resolve, reject) {
try {
self.recordRTC.resumeRecording(), resolve()
} catch (e) {
reject(e)
}
})
}, this.getDataURL = function(callback) {
return new Promise(function(resolve, reject) {
try {
self.recordRTC.getDataURL(function(dataURL) {
resolve(dataURL)
})
} catch (e) {
reject(e)
}
})
}, this.getBlob = function() {
return new Promise(function(resolve, reject) {
try {
resolve(self.recordRTC.getBlob())
} catch (e) {
reject(e)
}
})
}, this.getInternalRecorder = function() {
return new Promise(function(resolve, reject) {
try {
resolve(self.recordRTC.getInternalRecorder())
} catch (e) {
reject(e)
}
})
}, this.reset = function() {
return new Promise(function(resolve, reject) {
try {
resolve(self.recordRTC.reset())
} catch (e) {
reject(e)
}
})
}, this.destroy = function() {
return new Promise(function(resolve, reject) {
try {
resolve(self.recordRTC.destroy())
} catch (e) {
reject(e)
}
})
}, this.getState = function() {
return new Promise(function(resolve, reject) {
try {
resolve(self.recordRTC.getState())
} catch (e) {
reject(e)
}
})
}, this.blob = null, this.version = "5.6.2"
}
function WebAssemblyRecorder(stream, config) {
function cameraStream() {
return new ReadableStream({
start: function(controller) {
var cvs = document.createElement("canvas"),
video = document.createElement("video"),
first = !0;
video.srcObject = stream, video.muted = !0, video.height = config.height, video.width = config.width, video.volume = 0, video.onplaying = function() {
cvs.width = config.width, cvs.height = config.height;
var ctx = cvs.getContext("2d"),
frameTimeout = 1e3 / config.frameRate,
cameraTimer = setInterval(function() {
if (finished && (clearInterval(cameraTimer), controller.close()), first && (first = !1, config.onVideoProcessStarted && config.onVideoProcessStarted()), ctx.drawImage(video, 0, 0), "closed" !== controller
._controlledReadableStream.state) try {
controller.enqueue(ctx.getImageData(0, 0, config.width, config.height))
} catch (e) {}
}, frameTimeout)
}, video.play()
}
})
}
function startRecording(stream, buffer) {
if (!config.workerPath && !buffer) return finished = !1, void fetch("https://unpkg.com/webm-wasm@latest/dist/webm-worker.js").then(function(r) {
r.arrayBuffer().then(function(buffer) {
startRecording(stream, buffer)
})
});
if (!config.workerPath && buffer instanceof ArrayBuffer) {
var blob = new Blob([buffer], {
type: "text/javascript"
});
config.workerPath = URL.createObjectURL(blob)
}
config.workerPath || console.error("workerPath parameter is missing."), worker = new Worker(config.workerPath), worker.postMessage(config.webAssemblyPath || "https://unpkg.com/webm-wasm@latest/dist/webm-wasm.wasm"), worker
.addEventListener("message", function(event) {
"READY" === event.data ? (worker.postMessage({
width: config.width,
height: config.height,
bitrate: config.bitrate || 1200,
timebaseDen: config.frameRate || 30,
realtime: config.realtime
}), cameraStream().pipeTo(new WritableStream({
write: function(image) {
return finished ? void console.error("Got image, but recorder is finished!") : void worker.postMessage(image.data.buffer, [image.data.buffer])
}
}))) : event.data && (isPaused || arrayOfBuffers.push(event.data))
})
}
function terminate(callback) {
return worker ? (worker.addEventListener("message", function(event) {
null === event.data && (worker.terminate(), worker = null, callback && callback())
}), void worker.postMessage(null)) : void(callback && callback())
}
"undefined" != typeof ReadableStream && "undefined" != typeof WritableStream || console.error("Following polyfill is strongly recommended: https://unpkg.com/@mattiasbuelens/web-streams-polyfill/dist/polyfill.min.js"), config = config || {},
config.width = config.width || 640, config.height = config.height || 480, config.frameRate = config.frameRate || 30, config.bitrate = config.bitrate || 1200, config.realtime = config.realtime || !0;
var finished, worker;
this.record = function() {
arrayOfBuffers = [], isPaused = !1, this.blob = null, startRecording(stream), "function" == typeof config.initCallback && config.initCallback()
};
var isPaused;
this.pause = function() {
isPaused = !0
}, this.resume = function() {
isPaused = !1
};
var arrayOfBuffers = [];
this.stop = function(callback) {
finished = !0;
var recorder = this;
terminate(function() {
recorder.blob = new Blob(arrayOfBuffers, {
type: "video/webm"
}), callback(recorder.blob)
})
}, this.name = "WebAssemblyRecorder", this.toString = function() {
return this.name
}, this.clearRecordedData = function() {
arrayOfBuffers = [], isPaused = !1, this.blob = null
}, this.blob = null
}
RecordRTC.version = "5.6.2", "undefined" != typeof module && (module.exports = RecordRTC), "function" == typeof define && define.amd && define("RecordRTC", [], function() {
return RecordRTC
}), RecordRTC.getFromDisk = function(type, callback) {
if (!callback) throw "callback is mandatory.";
console.log("Getting recorded " + ("all" === type ? "blobs" : type + " blob ") + " from disk!"), DiskStorage.Fetch(function(dataURL, _type) {
"all" !== type && _type === type + "Blob" && callback && callback(dataURL), "all" === type && callback && callback(dataURL, _type.replace("Blob", ""))
})
}, RecordRTC.writeToDisk = function(options) {
console.log("Writing recorded blob(s) to disk!"), options = options || {}, options.audio && options.video && options.gif ? options.audio.getDataURL(function(audioDataURL) {
options.video.getDataURL(function(videoDataURL) {
options.gif.getDataURL(function(gifDataURL) {
DiskStorage.Store({
audioBlob: audioDataURL,
videoBlob: videoDataURL,
gifBlob: gifDataURL
})
})
})
}) : options.audio && options.video ? options.audio.getDataURL(function(audioDataURL) {
options.video.getDataURL(function(videoDataURL) {
DiskStorage.Store({
audioBlob: audioDataURL,
videoBlob: videoDataURL
})
})
}) : options.audio && options.gif ? options.audio.getDataURL(function(audioDataURL) {
options.gif.getDataURL(function(gifDataURL) {
DiskStorage.Store({
audioBlob: audioDataURL,
gifBlob: gifDataURL
})
})
}) : options.video && options.gif ? options.video.getDataURL(function(videoDataURL) {
options.gif.getDataURL(function(gifDataURL) {
DiskStorage.Store({
videoBlob: videoDataURL,
gifBlob: gifDataURL
})
})
}) : options.audio ? options.audio.getDataURL(function(audioDataURL) {
DiskStorage.Store({
audioBlob: audioDataURL
})
}) : options.video ? options.video.getDataURL(function(videoDataURL) {
DiskStorage.Store({
videoBlob: videoDataURL
})
}) : options.gif && options.gif.getDataURL(function(gifDataURL) {
DiskStorage.Store({
gifBlob: gifDataURL
})
})
}, MRecordRTC.getFromDisk = RecordRTC.getFromDisk, MRecordRTC.writeToDisk = RecordRTC.writeToDisk, "undefined" != typeof RecordRTC && (RecordRTC.MRecordRTC = MRecordRTC);
var browserFakeUserAgent = "Fake/5.0 (FakeOS) AppleWebKit/123 (KHTML, like Gecko) Fake/12.3.4567.89 Fake/123.45";
! function(that) {
that && "undefined" == typeof window && "undefined" != typeof global && (global.navigator = {
userAgent: browserFakeUserAgent,
getUserMedia: function() {}
}, global.console || (global.console = {}), "undefined" != typeof global.console.log && "undefined" != typeof global.console.error || (global.console.error = global.console.log = global.console.log || function() {
console.log(arguments)
}), "undefined" == typeof document && (that.document = {
documentElement: {
appendChild: function() {
return ""
}
}
}, document.createElement = document.captureStream = document.mozCaptureStream = function() {
var obj = {
getContext: function() {
return obj
},
play: function() {},
pause: function() {},
drawImage: function() {},
toDataURL: function() {
return ""
},
style: {}
};
return obj
}, that.HTMLVideoElement = function() {}), "undefined" == typeof location && (that.location = {
protocol: "file:",
href: "",
hash: ""
}), "undefined" == typeof screen && (that.screen = {
width: 0,
height: 0
}), "undefined" == typeof URL && (that.URL = {
createObjectURL: function() {
return ""
},
revokeObjectURL: function() {
return ""
}
}), that.window = global)
}("undefined" != typeof global ? global : null);
var requestAnimationFrame = window.requestAnimationFrame;
if ("undefined" == typeof requestAnimationFrame)
if ("undefined" != typeof webkitRequestAnimationFrame) requestAnimationFrame = webkitRequestAnimationFrame;
else if ("undefined" != typeof mozRequestAnimationFrame) requestAnimationFrame = mozRequestAnimationFrame;
else if ("undefined" != typeof msRequestAnimationFrame) requestAnimationFrame = msRequestAnimationFrame;
else if ("undefined" == typeof requestAnimationFrame) {
var lastTime = 0;
requestAnimationFrame = function(callback, element) {
var currTime = (new Date).getTime(),
timeToCall = Math.max(0, 16 - (currTime - lastTime)),
id = setTimeout(function() {
callback(currTime + timeToCall)
}, timeToCall);
return lastTime = currTime + timeToCall, id
}
}
var cancelAnimationFrame = window.cancelAnimationFrame;
"undefined" == typeof cancelAnimationFrame && ("undefined" != typeof webkitCancelAnimationFrame ? cancelAnimationFrame = webkitCancelAnimationFrame : "undefined" != typeof mozCancelAnimationFrame ? cancelAnimationFrame =
mozCancelAnimationFrame : "undefined" != typeof msCancelAnimationFrame ? cancelAnimationFrame = msCancelAnimationFrame : "undefined" == typeof cancelAnimationFrame && (cancelAnimationFrame = function(id) {
clearTimeout(id)
}));
var AudioContext = window.AudioContext;
"undefined" == typeof AudioContext && ("undefined" != typeof webkitAudioContext && (AudioContext = webkitAudioContext), "undefined" != typeof mozAudioContext && (AudioContext = mozAudioContext));
var URL = window.URL;
"undefined" == typeof URL && "undefined" != typeof webkitURL && (URL = webkitURL), "undefined" != typeof navigator && "undefined" == typeof navigator.getUserMedia && ("undefined" != typeof navigator.webkitGetUserMedia && (navigator
.getUserMedia = navigator.webkitGetUserMedia), "undefined" != typeof navigator.mozGetUserMedia && (navigator.getUserMedia = navigator.mozGetUserMedia));
var isEdge = !(navigator.userAgent.indexOf("Edge") === -1 || !navigator.msSaveBlob && !navigator.msSaveOrOpenBlob),
isOpera = !!window.opera || navigator.userAgent.indexOf("OPR/") !== -1,
isFirefox = navigator.userAgent.toLowerCase().indexOf("firefox") > -1 && "netscape" in window && / rv:/.test(navigator.userAgent),
isChrome = !isOpera && !isEdge && !!navigator.webkitGetUserMedia || isElectron() || navigator.userAgent.toLowerCase().indexOf("chrome/") !== -1,
isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
isSafari && !isChrome && navigator.userAgent.indexOf("CriOS") !== -1 && (isSafari = !1, isChrome = !0);
var MediaStream = window.MediaStream;
"undefined" == typeof MediaStream && "undefined" != typeof webkitMediaStream && (MediaStream = webkitMediaStream), "undefined" != typeof MediaStream && "undefined" == typeof MediaStream.prototype.stop && (MediaStream.prototype.stop =
function() {
this.getTracks().forEach(function(track) {
track.stop()
})
}), "undefined" != typeof RecordRTC && (RecordRTC.invokeSaveAsDialog = invokeSaveAsDialog, RecordRTC.getTracks = getTracks, RecordRTC.getSeekableBlob = getSeekableBlob, RecordRTC.bytesToSize = bytesToSize, RecordRTC.isElectron =
isElectron);
var Storage = {};
"undefined" != typeof AudioContext ? Storage.AudioContext = AudioContext : "undefined" != typeof webkitAudioContext && (Storage.AudioContext = webkitAudioContext), "undefined" != typeof RecordRTC && (RecordRTC.Storage = Storage),
"undefined" != typeof RecordRTC && (RecordRTC.MediaStreamRecorder = MediaStreamRecorder), "undefined" != typeof RecordRTC && (RecordRTC.StereoAudioRecorder = StereoAudioRecorder), "undefined" != typeof RecordRTC && (RecordRTC
.CanvasRecorder = CanvasRecorder), "undefined" != typeof RecordRTC && (RecordRTC.WhammyRecorder = WhammyRecorder);
var Whammy = function() {
function WhammyVideo(duration) {
this.frames = [], this.duration = duration || 1, this.quality = .8
}
function processInWebWorker(_function) {
var blob = URL.createObjectURL(new Blob([_function.toString(), "this.onmessage = function (eee) {" + _function.name + "(eee.data);}"], {
type: "application/javascript"
})),
worker = new Worker(blob);
return URL.revokeObjectURL(blob), worker
}
function whammyInWebWorker(frames) {
function ArrayToWebM(frames) {
var info = checkFrames(frames);
if (!info) return [];
for (var clusterMaxDuration = 3e4, EBML = [{
id: 440786851,
data: [{
data: 1,
id: 17030
}, {
data: 1,
id: 17143
}, {
data: 4,
id: 17138
}, {
data: 8,
id: 17139
}, {
data: "webm",
id: 17026
}, {
data: 2,
id: 17031
}, {
data: 2,
id: 17029
}]
}, {
id: 408125543,
data: [{
id: 357149030,
data: [{
data: 1e6,
id: 2807729
}, {
data: "whammy",
id: 19840
}, {
data: "whammy",
id: 22337
}, {
data: doubleToString(info.duration),
id: 17545
}]
}, {
id: 374648427,
data: [{
id: 174,
data: [{
data: 1,
id: 215
}, {
data: 1,
id: 29637
}, {
data: 0,
id: 156
}, {
data: "und",
id: 2274716
}, {
data: "V_VP8",
id: 134
}, {
data: "VP8",
id: 2459272
}, {
data: 1,
id: 131
}, {
id: 224,
data: [{
data: info.width,
id: 176
}, {
data: info.height,
id: 186
}]
}]
}]
}]
}], frameNumber = 0, clusterTimecode = 0; frameNumber < frames.length;) {
var clusterFrames = [],
clusterDuration = 0;
do clusterFrames.push(frames[frameNumber]), clusterDuration += frames[frameNumber].duration, frameNumber++; while (frameNumber < frames.length && clusterDuration < clusterMaxDuration);
var clusterCounter = 0,
cluster = {
id: 524531317,
data: getClusterData(clusterTimecode, clusterCounter, clusterFrames)
};
EBML[1].data.push(cluster), clusterTimecode += clusterDuration
}
return generateEBML(EBML)
}
function getClusterData(clusterTimecode, clusterCounter, clusterFrames) {
return [{
data: clusterTimecode,
id: 231
}].concat(clusterFrames.map(function(webp) {
var block = makeSimpleBlock({
discardable: 0,
frame: webp.data.slice(4),
invisible: 0,
keyframe: 1,
lacing: 0,
trackNum: 1,
timecode: Math.round(clusterCounter)
});
return clusterCounter += webp.duration, {
data: block,
id: 163
}
}))
}
function checkFrames(frames) {
if (!frames[0]) return void postMessage({
error: "Something went wrong. Maybe WebP format is not supported in the current browser."
});
for (var width = frames[0].width, height = frames[0].height, duration = frames[0].duration, i = 1; i < frames.length; i++) duration += frames[i].duration;
return {
duration: duration,
width: width,
height: height
}
}
function numToBuffer(num) {
for (var parts = []; num > 0;) parts.push(255 & num), num >>= 8;
return new Uint8Array(parts.reverse())
}
function strToBuffer(str) {
return new Uint8Array(str.split("").map(function(e) {
return e.charCodeAt(0)
}))
}
function bitsToBuffer(bits) {
var data = [],
pad = bits.length % 8 ? new Array(9 - bits.length % 8).join("0") : "";
bits = pad + bits;
for (var i = 0; i < bits.length; i += 8) data.push(parseInt(bits.substr(i, 8), 2));
return new Uint8Array(data)
}
function generateEBML(json) {
for (var ebml = [], i = 0; i < json.length; i++) {
var data = json[i].data;
"object" == typeof data && (data = generateEBML(data)), "number" == typeof data && (data = bitsToBuffer(data.toString(2))), "string" == typeof data && (data = strToBuffer(data));
var len = data.size || data.byteLength || data.length,
zeroes = Math.ceil(Math.ceil(Math.log(len) / Math.log(2)) / 8),
sizeToString = len.toString(2),
padded = new Array(7 * zeroes + 7 + 1 - sizeToString.length).join("0") + sizeToString,
size = new Array(zeroes).join("0") + "1" + padded;
ebml.push(numToBuffer(json[i].id)), ebml.push(bitsToBuffer(size)), ebml.push(data)
}
return new Blob(ebml, {
type: "video/webm"
})
}
function makeSimpleBlock(data) {
var flags = 0;
if (data.keyframe && (flags |= 128), data.invisible && (flags |= 8), data.lacing && (flags |= data.lacing << 1), data.discardable && (flags |= 1), data.trackNum > 127) throw "TrackNumber > 127 not supported";
var out = [128 | data.trackNum, data.timecode >> 8, 255 & data.timecode, flags].map(function(e) {
return String.fromCharCode(e)
}).join("") + data.frame;
return out
}
function parseWebP(riff) {
for (var VP8 = riff.RIFF[0].WEBP[0], frameStart = VP8.indexOf("*"), i = 0, c = []; i < 4; i++) c[i] = VP8.charCodeAt(frameStart + 3 + i);
var width, height, tmp;
return tmp = c[1] << 8 | c[0], width = 16383 & tmp, tmp = c[3] << 8 | c[2], height = 16383 & tmp, {
width: width,
height: height,
data: VP8,
riff: riff
}
}
function getStrLength(string, offset) {
return parseInt(string.substr(offset + 4, 4).split("").map(function(i) {
var unpadded = i.charCodeAt(0).toString(2);
return new Array(8 - unpadded.length + 1).join("0") + unpadded
}).join(""), 2)
}
function parseRIFF(string) {
for (var offset = 0, chunks = {}; offset < string.length;) {
var id = string.substr(offset, 4),
len = getStrLength(string, offset),
data = string.substr(offset + 4 + 4, len);
offset += 8 + len, chunks[id] = chunks[id] || [], "RIFF" === id || "LIST" === id ? chunks[id].push(parseRIFF(data)) : chunks[id].push(data)
}
return chunks
}
function doubleToString(num) {
return [].slice.call(new Uint8Array(new Float64Array([num]).buffer), 0).map(function(e) {
return String.fromCharCode(e)
}).reverse().join("")
}
var webm = new ArrayToWebM(frames.map(function(frame) {
var webp = parseWebP(parseRIFF(atob(frame.image.slice(23))));
return webp.duration = frame.duration, webp
}));
postMessage(webm)
}
return WhammyVideo.prototype.add = function(frame, duration) {
if ("canvas" in frame && (frame = frame.canvas), "toDataURL" in frame && (frame = frame.toDataURL("image/webp", this.quality)), !/^data:image\/webp;base64,/gi.test(frame))
throw "Input must be formatted properly as a base64 encoded DataURI of type image/webp";
this.frames.push({
image: frame,
duration: duration || this.duration
})
}, WhammyVideo.prototype.compile = function(callback) {
var webWorker = processInWebWorker(whammyInWebWorker);
webWorker.onmessage = function(event) {
return event.data.error ? void console.error(event.data.error) : void callback(event.data)
}, webWorker.postMessage(this.frames)
}, {
Video: WhammyVideo
}
}();
"undefined" != typeof RecordRTC && (RecordRTC.Whammy = Whammy);
var DiskStorage = {
init: function() {
function createObjectStore(dataBase) {
dataBase.createObjectStore(self.dataStoreName)
}
function putInDB() {
function getFromStore(portionName) {
transaction.objectStore(self.dataStoreName).get(portionName).onsuccess = function(event) {
self.callback && self.callback(event.target.result, portionName)
}
}
var transaction = db.transaction([self.dataStoreName], "readwrite");
self.videoBlob && transaction.objectStore(self.dataStoreName).put(self.videoBlob, "videoBlob"), self.gifBlob && transaction.objectStore(self.dataStoreName).put(self.gifBlob, "gifBlob"), self.audioBlob && transaction.objectStore(self
.dataStoreName).put(self.audioBlob, "audioBlob"), getFromStore("audioBlob"), getFromStore("videoBlob"), getFromStore("gifBlob")
}
var self = this;
if ("undefined" == typeof indexedDB || "undefined" == typeof indexedDB.open) return void console.error("IndexedDB API are not available in this browser.");
var db, dbVersion = 1,
dbName = this.dbName || location.href.replace(/\/|:|#|%|\.|\[|\]/g, ""),
request = indexedDB.open(dbName, dbVersion);
request.onerror = self.onError, request.onsuccess = function() {
if (db = request.result, db.onerror = self.onError, db.setVersion)
if (db.version !== dbVersion) {
var setVersion = db.setVersion(dbVersion);
setVersion.onsuccess = function() {
createObjectStore(db), putInDB()
}
} else putInDB();
else putInDB()
}, request.onupgradeneeded = function(event) {
createObjectStore(event.target.result)
}
},
Fetch: function(callback) {
return this.callback = callback, this.init(), this
},
Store: function(config) {
return this.audioBlob = config.audioBlob, this.videoBlob = config.videoBlob, this.gifBlob = config.gifBlob, this.init(), this
},
onError: function(error) {
console.error(JSON.stringify(error, null, "\t"))
},
dataStoreName: "recordRTC",
dbName: null
};
"undefined" != typeof RecordRTC && (RecordRTC.DiskStorage = DiskStorage), "undefined" != typeof RecordRTC && (RecordRTC.GifRecorder = GifRecorder), "undefined" == typeof RecordRTC && ("undefined" != typeof module && (module.exports =
MultiStreamsMixer), "function" == typeof define && define.amd && define("MultiStreamsMixer", [], function() {
return MultiStreamsMixer
})), "undefined" != typeof RecordRTC && (RecordRTC.MultiStreamRecorder = MultiStreamRecorder), "undefined" != typeof RecordRTC && (RecordRTC.RecordRTCPromisesHandler = RecordRTCPromisesHandler), "undefined" != typeof RecordRTC && (RecordRTC
.WebAssemblyRecorder = WebAssemblyRecorder);
'use strict';
// Last Updated On: 2020-08-12 11:18:41 AM UTC
// ________________
// DetectRTC v1.4.1
// Open-Sourced: https://github.com/muaz-khan/DetectRTC
// --------------------------------------------------
// Muaz Khan - www.MuazKhan.com
// MIT License - www.WebRTC-Experiment.com/licence
// --------------------------------------------------
"use strict";
! function() {
function getBrowserInfo() {
var nameOffset, verOffset, ix, nAgt = (navigator.appVersion, navigator.userAgent),
browserName = navigator.appName,
fullVersion = "" + parseFloat(navigator.appVersion),
majorVersion = parseInt(navigator.appVersion, 10);
if (isOpera) {
browserName = "Opera";
try {
fullVersion = navigator.userAgent.split("OPR/")[1].split(" ")[0], majorVersion = fullVersion.split(".")[0]
} catch (e) {
fullVersion = "0.0.0.0", majorVersion = 0
}
} else isIE ? (verOffset = nAgt.indexOf("rv:"), verOffset > 0 ? fullVersion = nAgt.substring(verOffset + 3) : (verOffset = nAgt.indexOf("MSIE"), fullVersion = nAgt.substring(verOffset + 5)), browserName = "IE") : isChrome ? (verOffset =
nAgt.indexOf("Chrome"), browserName = "Chrome", fullVersion = nAgt.substring(verOffset + 7)) : isSafari ? nAgt.indexOf("CriOS") !== -1 ? (verOffset = nAgt.indexOf("CriOS"), browserName = "Chrome", fullVersion = nAgt.substring(
verOffset + 6)) : nAgt.indexOf("FxiOS") !== -1 ? (verOffset = nAgt.indexOf("FxiOS"), browserName = "Firefox", fullVersion = nAgt.substring(verOffset + 6)) : (verOffset = nAgt.indexOf("Safari"), browserName = "Safari", fullVersion =
nAgt.substring(verOffset + 7), (verOffset = nAgt.indexOf("Version")) !== -1 && (fullVersion = nAgt.substring(verOffset + 8)), navigator.userAgent.indexOf("Version/") !== -1 && (fullVersion = navigator.userAgent.split("Version/")[1]
.split(" ")[0])) : isFirefox ? (verOffset = nAgt.indexOf("Firefox"), browserName = "Firefox", fullVersion = nAgt.substring(verOffset + 8)) : (nameOffset = nAgt.lastIndexOf(" ") + 1) < (verOffset = nAgt.lastIndexOf("/")) && (
browserName = nAgt.substring(nameOffset, verOffset), fullVersion = nAgt.substring(verOffset + 1), browserName.toLowerCase() === browserName.toUpperCase() && (browserName = navigator.appName));
return isEdge && (browserName = "Edge", fullVersion = navigator.userAgent.split("Edge/")[1]), (ix = fullVersion.search(/[; \)]/)) !== -1 && (fullVersion = fullVersion.substring(0, ix)), majorVersion = parseInt("" + fullVersion, 10),
isNaN(majorVersion) && (fullVersion = "" + parseFloat(navigator.appVersion), majorVersion = parseInt(navigator.appVersion, 10)), {
fullVersion: fullVersion,
version: majorVersion,
name: browserName,
isPrivateBrowsing: !1
}
}
function retry(isDone, next) {
var currentTrial = 0,
maxRetry = 50,
isTimeout = !1,
id = window.setInterval(function() {
isDone() && (window.clearInterval(id), next(isTimeout)), currentTrial++ > maxRetry && (window.clearInterval(id), isTimeout = !0, next(isTimeout))
}, 10)
}
function isIE10OrLater(userAgent) {
var ua = userAgent.toLowerCase();
if (0 === ua.indexOf("msie") && 0 === ua.indexOf("trident")) return !1;
var match = /(?:msie|rv:)\s?([\d\.]+)/.exec(ua);
return !!(match && parseInt(match[1], 10) >= 10)
}
function detectPrivateMode(callback) {
var isPrivate;
try {
if (window.webkitRequestFileSystem) window.webkitRequestFileSystem(window.TEMPORARY, 1, function() {
isPrivate = !1
}, function(e) {
isPrivate = !0
});
else if (window.indexedDB && /Firefox/.test(window.navigator.userAgent)) {
var db;
try {
db = window.indexedDB.open("test"), db.onerror = function() {
return !0
}
} catch (e) {
isPrivate = !0
}
"undefined" == typeof isPrivate && retry(function() {
return "done" === db.readyState
}, function(isTimeout) {
isTimeout || (isPrivate = !db.result)
})
} else if (isIE10OrLater(window.navigator.userAgent)) {
isPrivate = !1;
try {
window.indexedDB || (isPrivate = !0)
} catch (e) {
isPrivate = !0
}
} else if (window.localStorage && /Safari/.test(window.navigator.userAgent)) {
try {
window.localStorage.setItem("test", 1)
} catch (e) {
isPrivate = !0
}
"undefined" == typeof isPrivate && (isPrivate = !1, window.localStorage.removeItem("test"))
}
} catch (e) {
isPrivate = !1
}
retry(function() {
return "undefined" != typeof isPrivate
}, function(isTimeout) {
callback(isPrivate)
})
}
function detectDesktopOS() {
for (var cs, unknown = "-", nVer = navigator.appVersion, nAgt = navigator.userAgent, os = unknown, clientStrings = [{
s: "Chrome OS",
r: /CrOS/
}, {
s: "Windows 10",
r: /(Windows 10.0|Windows NT 10.0)/
}, {
s: "Windows 8.1",
r: /(Windows 8.1|Windows NT 6.3)/
}, {
s: "Windows 8",
r: /(Windows 8|Windows NT 6.2)/
}, {
s: "Windows 7",
r: /(Windows 7|Windows NT 6.1)/
}, {
s: "Windows Vista",
r: /Windows NT 6.0/
}, {
s: "Windows Server 2003",
r: /Windows NT 5.2/
}, {
s: "Windows XP",
r: /(Windows NT 5.1|Windows XP)/
}, {
s: "Windows 2000",
r: /(Windows NT 5.0|Windows 2000)/
}, {
s: "Windows ME",
r: /(Win 9x 4.90|Windows ME)/
}, {
s: "Windows 98",
r: /(Windows 98|Win98)/
}, {
s: "Windows 95",
r: /(Windows 95|Win95|Windows_95)/
}, {
s: "Windows NT 4.0",
r: /(Windows NT 4.0|WinNT4.0|WinNT|Windows NT)/
}, {
s: "Windows CE",
r: /Windows CE/
}, {
s: "Windows 3.11",
r: /Win16/
}, {
s: "Android",
r: /Android/
}, {
s: "Open BSD",
r: /OpenBSD/
}, {
s: "Sun OS",
r: /SunOS/
}, {
s: "Linux",
r: /(Linux|X11)/
}, {
s: "iOS",
r: /(iPhone|iPad|iPod)/
}, {
s: "Mac OS X",
r: /Mac OS X/
}, {
s: "Mac OS",
r: /(MacPPC|MacIntel|Mac_PowerPC|Macintosh)/
}, {
s: "QNX",
r: /QNX/
}, {
s: "UNIX",
r: /UNIX/
}, {
s: "BeOS",
r: /BeOS/
}, {
s: "OS/2",
r: /OS\/2/
}, {
s: "Search Bot",
r: /(nuhk|Googlebot|Yammybot|Openbot|Slurp|MSNBot|Ask Jeeves\/Teoma|ia_archiver)/
}], i = 0; cs = clientStrings[i]; i++)
if (cs.r.test(nAgt)) {
os = cs.s;
break
} var osVersion = unknown;
switch (/Windows/.test(os) && (/Windows (.*)/.test(os) && (osVersion = /Windows (.*)/.exec(os)[1]), os = "Windows"), os) {
case "Mac OS X":
/Mac OS X (10[\.\_\d]+)/.test(nAgt) && (osVersion = /Mac OS X (10[\.\_\d]+)/.exec(nAgt)[1]);
break;
case "Android":
/Android ([\.\_\d]+)/.test(nAgt) && (osVersion = /Android ([\.\_\d]+)/.exec(nAgt)[1]);
break;
case "iOS":
/OS (\d+)_(\d+)_?(\d+)?/.test(nAgt) && (osVersion = /OS (\d+)_(\d+)_?(\d+)?/.exec(nVer), osVersion && osVersion.length > 3 && (osVersion = osVersion[1] + "." + osVersion[2] + "." + (0 | osVersion[3])))
}
return {
osName: os,
osVersion: osVersion
}
}
function getAndroidVersion(ua) {
ua = (ua || navigator.userAgent).toLowerCase();
var match = ua.match(/android\s([0-9\.]*)/);
return !!match && match[1]
}
function DetectLocalIPAddress(callback, stream) {
if (DetectRTC.isWebRTCSupported) {
var isPublic = !0,
isIpv4 = !0;
getIPs(function(ip) {
ip ? ip.match(regexIpv4Local) ? (isPublic = !1, callback("Local: " + ip, isPublic, isIpv4)) : ip.match(regexIpv6) ? (isIpv4 = !1, callback("Public: " + ip, isPublic, isIpv4)) : callback("Public: " + ip, isPublic, isIpv4) :
callback()
}, stream)
}
}
function getIPs(callback, stream) {
function handleCandidate(candidate) {
if (!candidate) return void callback();
var match = regexIpv4.exec(candidate);
if (match) {
var ipAddress = match[1],
isPublic = candidate.match(regexIpv4Local),
isIpv4 = !0;
void 0 === ipDuplicates[ipAddress] && callback(ipAddress, isPublic, isIpv4), ipDuplicates[ipAddress] = !0
}
}
function afterCreateOffer() {
var lines = pc.localDescription.sdp.split("\n");
lines.forEach(function(line) {
line && 0 === line.indexOf("a=candidate:") && handleCandidate(line)
})
}
if ("undefined" != typeof document && "function" == typeof document.getElementById) {
var ipDuplicates = {},
RTCPeerConnection = window.RTCPeerConnection || window.mozRTCPeerConnection || window.webkitRTCPeerConnection;
if (!RTCPeerConnection) {
var iframe = document.getElementById("iframe");
if (!iframe) return;
var win = iframe.contentWindow;
RTCPeerConnection = win.RTCPeerConnection || win.mozRTCPeerConnection || win.webkitRTCPeerConnection
}
if (RTCPeerConnection) {
var peerConfig = null;
"Chrome" === DetectRTC.browser && DetectRTC.browser.version < 58 && (peerConfig = {
optional: [{
RtpDataChannels: !0
}]
});
var servers = {
iceServers: [{
urls: "stun:stun.l.google.com:19302"
}]
},
pc = new RTCPeerConnection(servers, peerConfig);
if (stream && (pc.addStream ? pc.addStream(stream) : pc.addTrack && stream.getTracks()[0] && pc.addTrack(stream.getTracks()[0], stream)), pc.onicecandidate = function(event) {
event.candidate && event.candidate.candidate ? handleCandidate(event.candidate.candidate) : handleCandidate()
}, !stream) try {
pc.createDataChannel("sctp", {})
} catch (e) {}
DetectRTC.isPromisesSupported ? pc.createOffer().then(function(result) {
pc.setLocalDescription(result).then(afterCreateOffer)
}) : pc.createOffer(function(result) {
pc.setLocalDescription(result, afterCreateOffer, function() {})
}, function() {})
}
}
}
function checkDeviceSupport(callback) {
if (!canEnumerate) return void(callback && callback());
if (!navigator.enumerateDevices && window.MediaStreamTrack && window.MediaStreamTrack.getSources && (navigator.enumerateDevices = window.MediaStreamTrack.getSources.bind(window.MediaStreamTrack)), !navigator.enumerateDevices && navigator
.enumerateDevices && (navigator.enumerateDevices = navigator.enumerateDevices.bind(navigator)), !navigator.enumerateDevices) return void(callback && callback());
MediaDevices = [], audioInputDevices = [], audioOutputDevices = [], videoInputDevices = [], hasMicrophone = !1, hasSpeakers = !1, hasWebcam = !1, isWebsiteHasMicrophonePermissions = !1, isWebsiteHasWebcamPermissions = !1;
var alreadyUsedDevices = {};
navigator.enumerateDevices(function(devices) {
MediaDevices = [], audioInputDevices = [], audioOutputDevices = [], videoInputDevices = [], devices.forEach(function(_device) {
var device = {};
for (var d in _device) try {
"function" != typeof _device[d] && (device[d] = _device[d])
} catch (e) {}
alreadyUsedDevices[device.deviceId + device.label + device.kind] || ("audio" === device.kind && (device.kind = "audioinput"), "video" === device.kind && (device.kind = "videoinput"), device.deviceId || (device.deviceId =
device.id), device.id || (device.id = device.deviceId), device.label ? ("videoinput" !== device.kind || isWebsiteHasWebcamPermissions || (isWebsiteHasWebcamPermissions = !0), "audioinput" !== device.kind ||
isWebsiteHasMicrophonePermissions || (isWebsiteHasMicrophonePermissions = !0)) : (device.isCustomLabel = !0, "videoinput" === device.kind ? device.label = "Camera " + (videoInputDevices.length + 1) : "audioinput" ===
device.kind ? device.label = "Microphone " + (audioInputDevices.length + 1) : "audiooutput" === device.kind ? device.label = "Speaker " + (audioOutputDevices.length + 1) : device.label =
"Please invoke getUserMedia once.", "undefined" != typeof DetectRTC && DetectRTC.browser.isChrome && DetectRTC.browser.version >= 46 && !/^(https:|chrome-extension:)$/g.test(location.protocol || "") && "undefined" !=
typeof document && "string" == typeof document.domain && document.domain.search && document.domain.search(/localhost|127.0./g) === -1 && (device.label = "HTTPs is required to get label of this " + device.kind +
" device.")), "audioinput" === device.kind && (hasMicrophone = !0, audioInputDevices.indexOf(device) === -1 && audioInputDevices.push(device)), "audiooutput" === device.kind && (hasSpeakers = !0, audioOutputDevices
.indexOf(device) === -1 && audioOutputDevices.push(device)), "videoinput" === device.kind && (hasWebcam = !0, videoInputDevices.indexOf(device) === -1 && videoInputDevices.push(device)), MediaDevices.push(device),
alreadyUsedDevices[device.deviceId + device.label + device.kind] = device)
}), "undefined" != typeof DetectRTC && (DetectRTC.MediaDevices = MediaDevices, DetectRTC.hasMicrophone = hasMicrophone, DetectRTC.hasSpeakers = hasSpeakers, DetectRTC.hasWebcam = hasWebcam, DetectRTC
.isWebsiteHasWebcamPermissions = isWebsiteHasWebcamPermissions, DetectRTC.isWebsiteHasMicrophonePermissions = isWebsiteHasMicrophonePermissions, DetectRTC.audioInputDevices = audioInputDevices, DetectRTC.audioOutputDevices =
audioOutputDevices, DetectRTC.videoInputDevices = videoInputDevices), callback && callback()
})
}
function getAspectRatio(w, h) {
function gcd(a, b) {
return 0 == b ? a : gcd(b, a % b)
}
var r = gcd(w, h);
return w / r / (h / r)
}
var browserFakeUserAgent = "Fake/5.0 (FakeOS) AppleWebKit/123 (KHTML, like Gecko) Fake/12.3.4567.89 Fake/123.45",
isNodejs = "object" == typeof process && "object" == typeof process.versions && process.versions.node && !process.browser;
if (isNodejs) {
var version = process.versions.node.toString().replace("v", "");
browserFakeUserAgent = "Nodejs/" + version + " (NodeOS) AppleWebKit/" + version + " (KHTML, like Gecko) Nodejs/" + version + " Nodejs/" + version
}! function(that) {
"undefined" == typeof window && ("undefined" == typeof window && "undefined" != typeof global ? (global.navigator = {
userAgent: browserFakeUserAgent,
getUserMedia: function() {}
}, that.window = global) : "undefined" == typeof window, "undefined" == typeof location && (that.location = {
protocol: "file:",
href: "",
hash: ""
}), "undefined" == typeof screen && (that.screen = {
width: 0,
height: 0
}))
}("undefined" != typeof global ? global : window);
var navigator = window.navigator;
"undefined" != typeof navigator ? ("undefined" != typeof navigator.webkitGetUserMedia && (navigator.getUserMedia = navigator.webkitGetUserMedia), "undefined" != typeof navigator.mozGetUserMedia && (navigator.getUserMedia = navigator
.mozGetUserMedia)) : navigator = {
getUserMedia: function() {},
userAgent: browserFakeUserAgent
};
var isMobileDevice = !!/Android|webOS|iPhone|iPad|iPod|BB10|BlackBerry|IEMobile|Opera Mini|Mobile|mobile/i.test(navigator.userAgent || ""),
isEdge = !(navigator.userAgent.indexOf("Edge") === -1 || !navigator.msSaveOrOpenBlob && !navigator.msSaveBlob),
isOpera = !!window.opera || navigator.userAgent.indexOf(" OPR/") >= 0,
isFirefox = navigator.userAgent.toLowerCase().indexOf("firefox") > -1 && "netscape" in window && / rv:/.test(navigator.userAgent),
isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent),
isChrome = !!window.chrome && !isOpera,
isIE = "undefined" != typeof document && !!document.documentMode && !isEdge,
isMobile = {
Android: function() {
return navigator.userAgent.match(/Android/i)
},
BlackBerry: function() {
return navigator.userAgent.match(/BlackBerry|BB10/i)
},
iOS: function() {
return navigator.userAgent.match(/iPhone|iPad|iPod/i)
},
Opera: function() {
return navigator.userAgent.match(/Opera Mini/i)
},
Windows: function() {
return navigator.userAgent.match(/IEMobile/i)
},
any: function() {
return isMobile.Android() || isMobile.BlackBerry() || isMobile.iOS() || isMobile.Opera() || isMobile.Windows()
},
getOsName: function() {
var osName = "Unknown OS";
return isMobile.Android() && (osName = "Android"), isMobile.BlackBerry() && (osName = "BlackBerry"), isMobile.iOS() && (osName = "iOS"), isMobile.Opera() && (osName = "Opera Mini"), isMobile.Windows() && (osName = "Windows"), osName
}
},
osName = "Unknown OS",
osVersion = "Unknown OS Version",
osInfo = detectDesktopOS();
osInfo && osInfo.osName && "-" != osInfo.osName ? (osName = osInfo.osName, osVersion = osInfo.osVersion) : isMobile.any() && (osName = isMobile.getOsName(), "Android" == osName && (osVersion = getAndroidVersion()));
var isNodejs = "object" == typeof process && "object" == typeof process.versions && process.versions.node;
"Unknown OS" === osName && isNodejs && (osName = "Nodejs", osVersion = process.versions.node.toString().replace("v", ""));
var isCanvasSupportsStreamCapturing = !1,
isVideoSupportsStreamCapturing = !1;
["captureStream", "mozCaptureStream", "webkitCaptureStream"].forEach(function(item) {
"undefined" != typeof document && "function" == typeof document.createElement && (!isCanvasSupportsStreamCapturing && item in document.createElement("canvas") && (isCanvasSupportsStreamCapturing = !0), !
isVideoSupportsStreamCapturing && item in document.createElement("video") && (isVideoSupportsStreamCapturing = !0))
});
var regexIpv4Local = /^(192\.168\.|169\.254\.|10\.|172\.(1[6-9]|2\d|3[01]))/,
regexIpv4 = /([0-9]{1,3}(\.[0-9]{1,3}){3})/,
regexIpv6 = /[a-f0-9]{1,4}(:[a-f0-9]{1,4}){7}/,
MediaDevices = [],
audioInputDevices = [],
audioOutputDevices = [],
videoInputDevices = [];
navigator.mediaDevices && navigator.mediaDevices.enumerateDevices && (navigator.enumerateDevices = function(callback) {
var enumerateDevices = navigator.mediaDevices.enumerateDevices();
enumerateDevices && enumerateDevices.then ? navigator.mediaDevices.enumerateDevices().then(callback)["catch"](function() {
callback([])
}) : callback([])
});
var canEnumerate = !1;
"undefined" != typeof MediaStreamTrack && "getSources" in MediaStreamTrack ? canEnumerate = !0 : navigator.mediaDevices && navigator.mediaDevices.enumerateDevices && (canEnumerate = !0);
var hasMicrophone = !1,
hasSpeakers = !1,
hasWebcam = !1,
isWebsiteHasMicrophonePermissions = !1,
isWebsiteHasWebcamPermissions = !1,
DetectRTC = window.DetectRTC || {};
DetectRTC.browser = getBrowserInfo(), detectPrivateMode(function(isPrivateBrowsing) {
DetectRTC.browser.isPrivateBrowsing = !!isPrivateBrowsing
}), DetectRTC.browser["is" + DetectRTC.browser.name] = !0, DetectRTC.osName = osName, DetectRTC.osVersion = osVersion;
var isWebRTCSupported = ("object" == typeof process && "object" == typeof process.versions && process.versions["node-webkit"], !1);
["RTCPeerConnection", "webkitRTCPeerConnection", "mozRTCPeerConnection", "RTCIceGatherer"].forEach(function(item) {
isWebRTCSupported || item in window && (isWebRTCSupported = !0)
}), DetectRTC.isWebRTCSupported = isWebRTCSupported, DetectRTC.isORTCSupported = "undefined" != typeof RTCIceGatherer;
var isScreenCapturingSupported = !1;
if (DetectRTC.browser.isChrome && DetectRTC.browser.version >= 35 ? isScreenCapturingSupported = !0 : DetectRTC.browser.isFirefox && DetectRTC.browser.version >= 34 ? isScreenCapturingSupported = !0 : DetectRTC.browser.isEdge && DetectRTC
.browser.version >= 17 ? isScreenCapturingSupported = !0 : "Android" === DetectRTC.osName && DetectRTC.browser.isChrome && (isScreenCapturingSupported = !0), (navigator.getDisplayMedia || navigator.mediaDevices && navigator.mediaDevices
.getDisplayMedia) && (isScreenCapturingSupported = !0), !/^(https:|chrome-extension:)$/g.test(location.protocol || "")) {
var isNonLocalHost = "undefined" != typeof document && "string" == typeof document.domain && document.domain.search && document.domain.search(/localhost|127.0./g) === -1;
isNonLocalHost && (DetectRTC.browser.isChrome || DetectRTC.browser.isEdge || DetectRTC.browser.isOpera) ? isScreenCapturingSupported = !1 : DetectRTC.browser.isFirefox && (isScreenCapturingSupported = !1)
}
DetectRTC.isScreenCapturingSupported = isScreenCapturingSupported;
var webAudio = {
isSupported: !1,
isCreateMediaStreamSourceSupported: !1
};
["AudioContext", "webkitAudioContext", "mozAudioContext", "msAudioContext"].forEach(function(item) {
webAudio.isSupported || item in window && (webAudio.isSupported = !0, window[item] && "createMediaStreamSource" in window[item].prototype && (webAudio.isCreateMediaStreamSourceSupported = !0))
}), DetectRTC.isAudioContextSupported = webAudio.isSupported, DetectRTC.isCreateMediaStreamSourceSupported = webAudio.isCreateMediaStreamSourceSupported;
var isRtpDataChannelsSupported = !1;
DetectRTC.browser.isChrome && DetectRTC.browser.version > 31 && (isRtpDataChannelsSupported = !0), DetectRTC.isRtpDataChannelsSupported = isRtpDataChannelsSupported;
var isSCTPSupportd = !1;
DetectRTC.browser.isFirefox && DetectRTC.browser.version > 28 ? isSCTPSupportd = !0 : DetectRTC.browser.isChrome && DetectRTC.browser.version > 25 ? isSCTPSupportd = !0 : DetectRTC.browser.isOpera && DetectRTC.browser.version >= 11 && (
isSCTPSupportd = !0), DetectRTC.isSctpDataChannelsSupported = isSCTPSupportd, DetectRTC.isMobileDevice = isMobileDevice;
var isGetUserMediaSupported = !1;
navigator.getUserMedia ? isGetUserMediaSupported = !0 : navigator.mediaDevices && navigator.mediaDevices.getUserMedia && (isGetUserMediaSupported = !0), DetectRTC.browser.isChrome && DetectRTC.browser.version >= 46 && !
/^(https:|chrome-extension:)$/g.test(location.protocol || "") && "undefined" != typeof document && "string" == typeof document.domain && document.domain.search && document.domain.search(/localhost|127.0./g) === -1 && (
isGetUserMediaSupported = "Requires HTTPs"), "Nodejs" === DetectRTC.osName && (isGetUserMediaSupported = !1), DetectRTC.isGetUserMediaSupported = isGetUserMediaSupported;
var displayResolution = "";
if (screen.width) {
var width = screen.width ? screen.width : "",
height = screen.height ? screen.height : "";
displayResolution += "" + width + " x " + height
}
DetectRTC.displayResolution = displayResolution, DetectRTC.displayAspectRatio = getAspectRatio(screen.width, screen.height).toFixed(2), DetectRTC.isCanvasSupportsStreamCapturing = isCanvasSupportsStreamCapturing, DetectRTC
.isVideoSupportsStreamCapturing = isVideoSupportsStreamCapturing, "Chrome" == DetectRTC.browser.name && DetectRTC.browser.version >= 53 && (DetectRTC.isCanvasSupportsStreamCapturing || (DetectRTC.isCanvasSupportsStreamCapturing =
"Requires chrome flag: enable-experimental-web-platform-features"), DetectRTC.isVideoSupportsStreamCapturing || (DetectRTC.isVideoSupportsStreamCapturing = "Requires chrome flag: enable-experimental-web-platform-features")), DetectRTC
.DetectLocalIPAddress = DetectLocalIPAddress, DetectRTC.isWebSocketsSupported = "WebSocket" in window && 2 === window.WebSocket.CLOSING, DetectRTC.isWebSocketsBlocked = !DetectRTC.isWebSocketsSupported, "Nodejs" === DetectRTC.osName && (
DetectRTC.isWebSocketsSupported = !0, DetectRTC.isWebSocketsBlocked = !1), DetectRTC.checkWebSocketsSupport = function(callback) {
callback = callback || function() {};
try {
var starttime, websocket = new WebSocket("wss://echo.websocket.org:443/");
websocket.onopen = function() {
DetectRTC.isWebSocketsBlocked = !1, starttime = (new Date).getTime(), websocket.send("ping")
}, websocket.onmessage = function() {
DetectRTC.WebsocketLatency = (new Date).getTime() - starttime + "ms", callback(), websocket.close(), websocket = null
}, websocket.onerror = function() {
DetectRTC.isWebSocketsBlocked = !0, callback()
}
} catch (e) {
DetectRTC.isWebSocketsBlocked = !0, callback()
}
}, DetectRTC.load = function(callback) {
callback = callback || function() {}, checkDeviceSupport(callback)
}, "undefined" != typeof MediaDevices ? DetectRTC.MediaDevices = MediaDevices : DetectRTC.MediaDevices = [], DetectRTC.hasMicrophone = hasMicrophone, DetectRTC.hasSpeakers = hasSpeakers, DetectRTC.hasWebcam = hasWebcam, DetectRTC
.isWebsiteHasWebcamPermissions = isWebsiteHasWebcamPermissions, DetectRTC.isWebsiteHasMicrophonePermissions = isWebsiteHasMicrophonePermissions, DetectRTC.audioInputDevices = audioInputDevices, DetectRTC.audioOutputDevices =
audioOutputDevices, DetectRTC.videoInputDevices = videoInputDevices;
var isSetSinkIdSupported = !1;
"undefined" != typeof document && "function" == typeof document.createElement && "setSinkId" in document.createElement("video") && (isSetSinkIdSupported = !0), DetectRTC.isSetSinkIdSupported = isSetSinkIdSupported;
var isRTPSenderReplaceTracksSupported = !1;
DetectRTC.browser.isFirefox && "undefined" != typeof mozRTCPeerConnection ? "getSenders" in mozRTCPeerConnection.prototype && (isRTPSenderReplaceTracksSupported = !0) : DetectRTC.browser.isChrome && "undefined" !=
typeof webkitRTCPeerConnection && "getSenders" in webkitRTCPeerConnection.prototype && (isRTPSenderReplaceTracksSupported = !0), DetectRTC.isRTPSenderReplaceTracksSupported = isRTPSenderReplaceTracksSupported;
var isRemoteStreamProcessingSupported = !1;
DetectRTC.browser.isFirefox && DetectRTC.browser.version > 38 && (isRemoteStreamProcessingSupported = !0), DetectRTC.isRemoteStreamProcessingSupported = isRemoteStreamProcessingSupported;
var isApplyConstraintsSupported = !1;
"undefined" != typeof MediaStreamTrack && "applyConstraints" in MediaStreamTrack.prototype && (isApplyConstraintsSupported = !0), DetectRTC.isApplyConstraintsSupported = isApplyConstraintsSupported;
var isMultiMonitorScreenCapturingSupported = !1;
DetectRTC.browser.isFirefox && DetectRTC.browser.version >= 43 && (isMultiMonitorScreenCapturingSupported = !0), DetectRTC.isMultiMonitorScreenCapturingSupported = isMultiMonitorScreenCapturingSupported, DetectRTC.isPromisesSupported = !!(
"Promise" in window), DetectRTC.version = "1.4.1", "undefined" == typeof DetectRTC && (window.DetectRTC = {});
var MediaStream = window.MediaStream;
"undefined" == typeof MediaStream && "undefined" != typeof webkitMediaStream && (MediaStream = webkitMediaStream), "undefined" != typeof MediaStream && "function" == typeof MediaStream ? DetectRTC.MediaStream = Object.keys(MediaStream
.prototype) : DetectRTC.MediaStream = !1, "undefined" != typeof MediaStreamTrack ? DetectRTC.MediaStreamTrack = Object.keys(MediaStreamTrack.prototype) : DetectRTC.MediaStreamTrack = !1;
var RTCPeerConnection = window.RTCPeerConnection || window.mozRTCPeerConnection || window.webkitRTCPeerConnection;
"undefined" != typeof RTCPeerConnection ? DetectRTC.RTCPeerConnection = Object.keys(RTCPeerConnection.prototype) : DetectRTC.RTCPeerConnection = !1, window.DetectRTC = DetectRTC, "undefined" != typeof module && (module.exports = DetectRTC),
"function" == typeof define && define.amd && define("DetectRTC", [], function() {
return DetectRTC
})
}();
</script>
<script>
//
// Is there anything preventing a recording?
DetectRTC.load(function() {
var has_rtc = DetectRTC.isWebRTCSupported;
var has_webcam = DetectRTC.hasWebcam;
var has_mic = DetectRTC.hasMicrophone;
if ((has_webcam === false && has_mic === false) || has_rtc === false) {
//
// We are unable to record at all.
$('#message_record_container').addClass('hidden');
} else {
//
// We support at least one of the recording features.
$('#message_record_container').removeClass('hidden');
if (has_webcam === false) {
$('#record_video_audio').addClass('hidden');
}
if (has_mic === false) {
$('#record_audio').addClass('hidden');
}
}
});
//
// Now our actual recording.
const recording = document.querySelector('#recording_element');
let recorder = null;
var type;
var visualiser_timer;
var animation_duration = 2;
var visualiser_count = 0;
var animation_done = false;
document.getElementById('record_audio').addEventListener("click", function() {
type = 'audio';
startRecording('audio');
});
document.getElementById('record_video_audio').addEventListener("click", function() {
type = 'video';
startRecording('video');
});
function visualiserCount() {
visualiser_timer = setInterval(function() {
if (visualiser_count < animation_duration) {
visualiser_count++;
} else {
if (animation_done == false) {
visualiser_count = 0;
}
}
}, 1000)
}
function BestFormat(format_wanted) {
if (format_wanted == 'video') {
var webm_support = MediaRecorder.isTypeSupported('video/webm');
var mp4_support = MediaRecorder.isTypeSupported('video/mp4');
var format = 'video/webm';
format = (mp4_support === true) ? 'video/mp4' : format;
format = (webm_support === true) ? 'video/webm' : format;
} else {
var webm_support = MediaRecorder.isTypeSupported('audio/webm');
var mp4_support = MediaRecorder.isTypeSupported('audio/mp4');
var format = 'audio/webm';
format = (mp4_support === true) ? 'audio/mp4' : format;
format = (webm_support === true) ? 'audio/webm' : format;
}
return format;
}
async function startRecording() {
//
// Hide our main controls.
// $('#record_video_audio').addClass('hidden');
// $('#record_audio').addClass('hidden');
//
// What type of recording are we doing?
$('.recording-preview-text').addClass("hidden");
$('#recording_container').children("video").removeAttr('controls');
$('#record_audio').children("video").removeAttr('controls');
$('#close-window-option').attr("onclick", "CloseStartedRecording();");
$('.completed-voice-option').addClass("hidden");
$('.completed-video-option').addClass("hidden");
if (type == 'video') {
var recording_parms = {
video: {
facingMode: 'user'
},
audio: true
};
} else {
var recording_parms = {
audio: true
};
}
//
// Now request permission and start recording.
$('#recording_element').removeClass('hidden');
// $('#stop-option').removeClass('hidden');
$('.play-option').addClass("hidden");
visualiserCount();
let stream = await navigator.mediaDevices.getUserMedia(recording_parms).then(function(stream) {
if (type == 'video') {
$('#stop-option').removeClass('hidden');
$('.video-error').addClass("hidden");
$('#recording_container').appendTo('#video-preview');
$('.video-message').addClass("active");
$('#stop-option').appendTo('#video-finish-option');
$('.video-option').addClass("hidden").children("text").text("Re-Record");
$('.back-option').addClass("hidden");
$('.recording-preview').removeClass("hidden");
var element = document.getElementById('recording_element');
element.muted = "muted";
element.play();
} else {
$('#stop-option').removeClass('hidden');
$('.voice-error').addClass("hidden");
$('#recording_container').appendTo('#voice-preview');
$('#stop-option').appendTo('#voice-finish-option');
$('.voice-option').addClass("hidden").children("text").text("Re-Record");
$('.back-option').addClass("hidden");
$('.visualiser').removeClass("completed").addClass("active");
var element = document.getElementById('recording_element');
element.muted = "muted";
element.play();
}
recordingHandler(stream);
}).catch(function(err) {
if (type == 'video') {
$('.video-error').removeClass("hidden");
} else {
$('.voice-error').removeClass("hidden");
}
});
async function recordingHandler(stream) {
recording.srcObject = stream;
recorder = new RecordRTCPromisesHandler(stream, {
type: type,
// mimeType: (type == 'video' ? 'video/mp4' : 'audio/webm'),
mimeType: BestFormat(type),
audioBitsPerSecond: 128000,
videoBitsPerSecond: 512000,
recorderType: MediaStreamRecorder,
disableLogs: true,
});
recorder.recordRTC.setRecordingDuration(60000, async function() {
StopRecording();
await recorder.stopRecording();
stopRecordingCallback();
});
await recorder.startRecording();
recorder.stream = stream;
const internalRecorder = await recorder.getInternalRecorder();
};
};
//
// What to do when we stop recording.
// document.getElementById('record_stop').onclick = StopRecording;
document.getElementById('record_stop').addEventListener("click", function() {
StopRecording();
});
async function StopRecording() {
if (recorder !== null) {
if (typeof recorder.getState === 'function') {
var is_recording = await recorder.getState();
if (is_recording == 'recording') {
await recorder.stopRecording();
}
await stopRecordingCallback(type);
}
}
$('#recording_container').children("video").attr('controls', 'controls');
$('.video-message').removeClass("active");
$('#stop-option').addClass('hidden');
$('#record_video_audio').removeClass('hidden');
$('#record_audio').removeClass('hidden');
$('.video-option').removeClass("hidden");
$('.voice-option').removeClass("hidden");
$('.back-option').removeClass("hidden");
$('.completed-voice-option').removeClass("hidden");
$('.completed-video-option').removeClass("hidden");
$('.recording-preview').addClass("hidden");
var element = document.getElementById('recording_element');
element.muted = false;
animation_done = true;
var visualiser_closed = setInterval(function() {
if (visualiser_count == animation_duration) {
$('.visualiser').removeClass("active").addClass("completed");
$('.play-option').removeClass("hidden");
clearInterval(visualiser_closed);
}
}, 1000);
};
//
// Handle the save.
async function stopRecordingCallback() {
$('#submit_form').addClass('hidden');
recording.srcObject = null;
let blob = await recorder.getBlob();
var blob_url = URL.createObjectURL(blob);
recording.src = blob_url;
recorder.stream.getTracks().forEach(t => t.stop());
let data_url = await recorder.getDataURL();
var datas = {
'type': type,
'content': data_url,
};
$('#submit').attr("disabled", "true");
const ret = await psLoad('', 'save_recorded_message', datas);
var parsed = JSON.parse(ret);
$('#attr_videomsg').val(parsed.location);
$('#submit_form').removeClass('hidden');
if (quiet == false) {
$('.video-message-notification').removeClass("hidden");
setTimeout(function() {
$('.video-message-notification').addClass("hidden");
}, 2000);
}
$('#submit').removeAttr("disabled");
await recorder.reset();
await recorder.destroy();
recorder = null;
}
</script>
<input type="hidden" id="attr_videomsg" name="attr_videomsg" value="">
<div class="video-message-notification hidden">
<div class="container">
<div class="notification">
<i class="fa-solid fa-folder-arrow-up"></i>
<p>Your recording has been saved and a copy will be sent with this product.</p>
</div>
</div>
</div>
<table class="message-card-seperator" width="100%">
<tbody>
<tr>
<td>
<hr style="background-color: #eee;">
</td>
<td style="width: 1px; padding: 0 0px; white-space: nowrap;">Or</td>
<td>
<hr style="background-color: #eee;">
</td>
</tr>
</tbody>
</table>
<!--Standard Card-->
<div class="col-xs-12">
<div class="standard_card">
<label for="standard" style="display:inline">
<div class="checkmark-container">
<input class="ignore non-pretty" checked="checked" type="radio" name="attr_card_type" id="standard" value="standard">
<p>Message Card</p>
</div>
</label>
<script>
$(document).ready(function() {
var text_max = '200';
$('#count_attr_messagecard').html(text_max + ' Characters Remaining');
$('#attr_messagecard').keyup(function() {
var text_length = $('#attr_messagecard').val().length;
var text_remaining = text_max - text_length;
$('#count_attr_messagecard').html(text_remaining + ' Characters Remaining');
});
});
</script>
<label for="attr_messagecard">
<span style="font-size: 0;">Standard Message Card</span> <textarea placeholder="" rows="4" maxlength="200" id="attr_messagecard" class="form-control" name="attr_messagecard" onkeyup="AddonCrossSell($(this).val());"></textarea>
</label>
<div id="count_attr_messagecard">200 Characters Remaining</div>
<div class="clearfix"></div>
</div>
</div>
<div class="clearfix"></div>
</div>
<script>
// Timer for when user stops typing
var typingTimer;
var doneTypingInterval = 400;
$("#attr_messagecard").on('keyup', function() {
clearTimeout(typingTimer);
typingTimer = setTimeout(GetSuggestedAddons, doneTypingInterval);
});
$("#attr_messagecard").on('keydown', function() {
clearTimeout(typingTimer);
});
var images_url = 'https://images.prestigeflowers.co.uk/';
var site_url = 'https://www.prestigeflowers.co.uk/'
// Get suggested addons function
function GetSuggestedAddons() {
// Get the occasions
var occasions = [{
"id": 12,
"occasion": "Anniversary"
}, {
"id": 11,
"occasion": "Birthday"
}, {
"id": 13,
"occasion": "Congratulations"
}, {
"id": 8,
"occasion": "Graduation"
}, {
"id": 7,
"occasion": "Thank"
}];
// Get the message
var message = $('#attr_messagecard').val();
// If the message is not empty
if (message.length != 0) {
// See if any of the occasions are in the message
var matched_occasions = [];
for (var i = 0; i < occasions.length; i++) {
if (message.toLowerCase().indexOf(occasions[i].occasion.toLowerCase()) > -1) {
matched_occasions.push(occasions[i].occasion);
}
}
// If there are any matched occasions
if (matched_occasions.length != 0) {
// Get the suggested addons
psLoad('', 'suggested_addons', {
occasions: matched_occasions
}).then(function(response) {
response = JSON.parse(response);
if (response.success) {
// Define the addons variable
;
// Get included addons
var included_addons = {
"1337": {
"id": 34,
"sort": 0,
"addon_name": "Premium Feed",
"pretty_name": "Premium Feed",
"image_name": "premium-flower-food.webp",
"price": 0,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9716
}
};
// Get all addons allowed
var addons = [{
"id": 1007,
"sort": 0,
"addon_name": "180g Pralines",
"pretty_name": "Luxury Chocolates 180g",
"image_name": "PF_(AddOn)_Chocolate-RedPraline.jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "Delicious praline chocolates, perfect for any occasion, may contain nuts*",
"xsell_img": "pf-popup-updated-ayr.png",
"is_customer_specific": "No",
"stock": 1438
}, {
"id": 1008,
"sort": 0,
"addon_name": "400g Pralines",
"pretty_name": "Deluxe Chocolates 400g",
"image_name": "PF_(AddOn)_Chocolate-BlackPraline.jpg",
"price": 9.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "A box of delicious luxury praline chocolates, may contain nuts*",
"xsell_img": "PF-val-popup-chocs_2.png",
"is_customer_specific": "No",
"stock": 1592
}, {
"id": 5,
"sort": 0,
"addon_name": "Clear Vase",
"pretty_name": "Clear Vase",
"image_name": "new-clear-vase.png",
"price": 4.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "A clear and versatile glass vase perfect for this bouquet",
"xsell_img": "vase.jpg",
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 1049,
"sort": 0,
"addon_name": "PP2202",
"pretty_name": "Afternoon Tea Hamper",
"image_name": "aft-tea-postal-ENJOY.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "Delicious treats hamper perfect for celebrating",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 7019
}, {
"id": 1058,
"sort": 0,
"addon_name": "Guylian Chocolates",
"pretty_name": "Guylian Chocolates",
"image_name": "PF_(AddOn)_Chocolate-guylian.jpg",
"price": 2.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "Delicious Guylian chocolates the perfect accompaniment with flower. Weighs 42g",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 4696
}, {
"id": 1055,
"sort": 0,
"addon_name": "Birthday Cake",
"pretty_name": "Birthday Cake",
"image_name": "PF_(AddOn)_Gift-BirthdayCake.jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "A delicious hand made birthday cake by Valley Bakery.",
"xsell_img": "birthday-cake-extra.jpg",
"is_customer_specific": "No",
"stock": 6839
}, {
"id": 1051,
"sort": 0,
"addon_name": "Cuddly Bear",
"pretty_name": "Luxury Teddy Bear",
"image_name": "PF_(AddOn)_Bear.jpg",
"price": 9.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "Limited Edition handmade teddy ready for his new home!\n",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 5652
}, {
"id": 1072,
"sort": 0,
"addon_name": " Premium Assorted Belgian Chocolates - Round Box",
"pretty_name": "Premium Belgian Chocolates",
"image_name": "PF_(AddOn)_Chocolate-LuxuryRound.jpg",
"price": 19.99,
"distributor": "julian",
"global_feature": 1,
"enabled": 1,
"tooltip": "Delicious hand-crafted Belgian chocolates, perfect for elevating any occasion, this product contains alcohol",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 6998
}, {
"id": 1159,
"sort": 0,
"addon_name": "Just For You Add On Chocolate ",
"pretty_name": "Just For You Chocolate Bar",
"image_name": "PF_JustForYou-60gChocBar_AddOn(600px-300dpi).jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Delicious handcrafted Milk Chocolate 'Just For You' Valley Chocolate Bar (60g)",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 1121,
"sort": 0,
"addon_name": "Tea And Biscuits",
"pretty_name": "Tea and Biscuits",
"image_name": "new tea and biscuits.jpg",
"price": 3.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Delicious flower shaped biscuits with Yorkshire Tea Bags",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 61,
"sort": 0,
"addon_name": "Sparkling French Wine",
"pretty_name": "Sparkling French Wine (75cl)",
"image_name": "PF_(AddOn)_Drinks-Pigalle.jpg",
"price": 19.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "An elegant and refined bottle of bubbly",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 8797
}, {
"id": 1113,
"sort": 0,
"addon_name": "Happy Birthday Postal",
"pretty_name": "Happy Birthday Gift Box",
"image_name": "PF_(AddOn)_PostalHamper-Birthday-Birthday.jpg",
"price": 18.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "The ultimate Happy Birthday gift with cake, candles & treats!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9431
}, {
"id": 1158,
"sort": 0,
"addon_name": "Add On Happy Birthday Chocolate ",
"pretty_name": "Birthday Chocolate Bar",
"image_name": "PF_HappyBirthday-60gChocBar_AddOn(600px-300dpi).jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Delicious handcrafted Milk Chocolate 'Happy Birthday' Valley Chocolate Bar (60g)",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 1135,
"sort": 0,
"addon_name": "Spring Yankee Candle",
"pretty_name": "Garden Blooms Candle",
"image_name": "PF_(AddOn)_Candle-YankeeGardenBlooms.jpg",
"price": 9.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A beautiful Garden Blooms Yankee Candle, for all occasions!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 8934
}, {
"id": 1116,
"sort": 0,
"addon_name": "Afternoon Tea Gardening Postal",
"pretty_name": "Afternoon Tea Gardening Gift",
"image_name": "postal addons_0009_PH-Sent-with-love.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A lovely unique gift for nature lovers, artisan cakes, tea & seasonal planting bulbs! ",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9719
}, {
"id": 1076,
"sort": 0,
"addon_name": "Daffodil Bulbs",
"pretty_name": "Daffodil Bulbs",
"image_name": "PF--Daffodils-bulb-add-on (1).jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "10 Daffodil Bulbs ready for planting!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 1048,
"sort": 0,
"addon_name": "PP2208",
"pretty_name": "Wine Hamper",
"image_name": "wine-postal-ENJOY.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Cheers to all occasions with this wine gift!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9801
}, {
"id": 1155,
"sort": 0,
"addon_name": "New Champagne 75cl Add On",
"pretty_name": "Champagne 75cl",
"image_name": "PF_(AddOn)_Drinks-BaronVilleboerg.jpg",
"price": 34.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Luxurious Baron De Villeboerg Champagne perfect for celebrating!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9839
}, {
"id": 1157,
"sort": 0,
"addon_name": "GIN + TONIC CANDLE ADD ON",
"pretty_name": "Gin + Tonic Candle",
"image_name": "PF--Gin-and-Tonic-candle-add-on.jpg",
"price": 5.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A beautiful Gin & Tonic scented candle, for all occasions!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 13,
"sort": 0,
"addon_name": "White Wine 37.5cl",
"pretty_name": "White Wine (37.5cl)",
"image_name": "PF_(AddOn)_Drinks-WhiteWine37cl.jpg",
"price": 7.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Smooth, aromatic and refreshing French white wine, perfect to say Cheers! ",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9591
}, {
"id": 1039,
"sort": 0,
"addon_name": "Eco Clear Vase",
"pretty_name": "Eco Clear Vase",
"image_name": "luxe-vase.jpg",
"price": 8.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "New! Eco handmade vase made from recycled glass\n",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9559
}, {
"id": 1089,
"sort": 0,
"addon_name": "Dark Chocolate 90g Valley Chocolate ",
"pretty_name": "Dark Chocolate & Raspberry Bar",
"image_name": "PF_(AddOn)_Chocolate-DarkRaspberryBar.jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Delicious handcrafted Dark Chocolate and Raspberry Valley Chocolate Bar (90g)",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9644
}, {
"id": 1017,
"sort": 0,
"addon_name": "Birthday Pick",
"pretty_name": "Happy Birthday Pick",
"image_name": "birthday_pick.png",
"price": 2.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Add a delightful pick for extra surprise\n",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9689
}, {
"id": 14,
"sort": 0,
"addon_name": "Red Wine 37.5cl",
"pretty_name": "Red Wine (37.5cl)",
"image_name": "PF_(AddOn)_Drinks-RedWine37cl.jpg",
"price": 7.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A rich and full bodied delicious French red wine perfect to say Cheers! ",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9661
}, {
"id": 1140,
"sort": 0,
"addon_name": "Happy Birthday Candle Add On 2024",
"pretty_name": "Happy Birthday Candle ",
"image_name": "PF_(AddOn)_Candle-HappyBirthday.jpg",
"price": 2.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A colourful Happy Birthday candle to make their day extra special!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9935
}, {
"id": 1117,
"sort": 0,
"addon_name": "Lullaby Baby Postal",
"pretty_name": "New Baby Gift",
"image_name": "PF_(AddOn)_PostalHamper-NewBaby-NewBaby.jpg",
"price": 24.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "The perfect gift to celebrate a new baby with lovely lamb toys!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9954
}, {
"id": 1044,
"sort": 0,
"addon_name": "PP2204",
"pretty_name": "Perfect Pamper Gift",
"image_name": "pamper-hamper-ENJOY.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A delightful pamper hamper filled with goodies!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9552
}, {
"id": 111,
"sort": 0,
"addon_name": "Hand Blown Vase",
"pretty_name": "Hand Blown Vase",
"image_name": "hand_blown_vase.png",
"price": 9.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Handmade artisan vase, each unique\n",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9862
}, {
"id": 1154,
"sort": 0,
"addon_name": "Lemon Slices - Valley Sweets",
"pretty_name": "Fizzy Lemon Sweets",
"image_name": "PF_(AddOn)_Sweets-FizzyLemon.jpg",
"price": 3.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Delicious fizzy lemon sweet slices by Valley Sweets",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9960
}, {
"id": 1167,
"sort": 0,
"addon_name": "PP2206",
"pretty_name": "Chocolate Gift Box Gift",
"image_name": "6eb9b3858437161c45549380690e1975.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 1141,
"sort": 0,
"addon_name": "Moet 75cl",
"pretty_name": "Moet 75cl",
"image_name": "PF-MOET-75.CL.jpg",
"price": 65,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "The perfect extra for special occasions, delightful bottle of Moet 75cl Champagne",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9857
}, {
"id": 1046,
"sort": 0,
"addon_name": "Scented Occasion Candle",
"pretty_name": "Occasion Candle",
"image_name": "occasion-candle.png",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Beautifully frangrant candle, florist selected for the occasion\n",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9788
}, {
"id": 1077,
"sort": 0,
"addon_name": "Tulip Bulbs Queen of the Night",
"pretty_name": "Tulip Planting Bulbs",
"image_name": "PF-Tulip-bulb-addon (1).jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "10 Tulip Bulbs ready for planting!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9999
}, {
"id": 15,
"sort": 0,
"addon_name": "Moet 37.5cl",
"pretty_name": "Moet (37.5cl)",
"image_name": "half-moet.jpg",
"price": 39.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "The perfect extra gift to make any occasion that much more extraordinary",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9746
}, {
"id": 58,
"sort": 0,
"addon_name": "Gin 5cl + Tonic",
"pretty_name": "Gin 5cl + Tonic",
"image_name": "PF_(AddOn)_Drinks-GinTonic.jpg",
"price": 4.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A classic cocktail drink perfect for celebrating ",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9576
}, {
"id": 1145,
"sort": 0,
"addon_name": "PP2202 Happy Birthday Postcard",
"pretty_name": "Happy Birthday Afternoon Tea",
"image_name": "PF_(AddOn)_PostalHamper-AfternoonTea-Birthday.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "Delectable Afternoon Tea postal gift, perfect for all occasions! ",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9881
}, {
"id": 1043,
"sort": 0,
"addon_name": "PP2205",
"pretty_name": "Gin and Tonic Gift",
"image_name": "PH-Gin-2024-Enjoy.jpg",
"price": 16.99,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "A beautiful hamper for gin & tonic lovers!",
"xsell_img": null,
"is_customer_specific": "No",
"stock": 9986
}, {
"id": 34,
"sort": 0,
"addon_name": "Premium Feed",
"pretty_name": "Premium Feed",
"image_name": "premium-flower-food.webp",
"price": 0,
"distributor": "julian",
"global_feature": 0,
"enabled": 1,
"tooltip": "",
"xsell_img": "",
"is_customer_specific": "No",
"stock": 9716
}]
var addons = addons.reduce(function(map, obj) {
map[obj.id] = obj;
return map;
}, {});
// Get the suggested addons
var suggested_addons = response.suggested_addons;
// Loop through suggested addons and find the addon info
var matched_addons = [];
for (var i = 0; i < suggested_addons.length; i++) {
var addon = addons[suggested_addons[i]];
matched_addons.push(addon);
}
// Get a list of currently selected addons
var selected_addons = [];
$('input[name^="attr_addons_"]').each(function() {
if ($(this).prop('checked') == true) {
selected_addons.push($(this).attr('name'));
}
});
// Check if any of the matched addons are undefined and remove them
for (var i = matched_addons.length - 1; i >= 0; --i) {
if (matched_addons[i] == undefined) {
matched_addons.splice(i, 1);
}
}
// if there are any matched addons
if (matched_addons.length != 0) {
// Only get the first 5 addons
matched_addons = matched_addons.slice(0, 4);
// Foreach matched_addon add it to the .suggested-addons
var html = '';
for (var i = 0; i < matched_addons.length; i++) {
// Remove spaces from addon_name
var new_addon_name = matched_addons[i].addon_name.replace(/\s/g, '');
// Set defaults for checked, readonly and price
var checked = '';
var readonly = '';
var price = '£' + parseFloat(matched_addons[i].price).toFixed(2);
// Set the select_addon variable
var select_addon = 'attr_addons_' + matched_addons[i].id + '_' + matched_addons[i].addon_name;
// If the addon is included
if (included_addons[matched_addons[i].id] != undefined) {
checked = 'checked';
readonly = 'disabled';
price = 'Included!';
}
// If the addon is already selected
if (selected_addons.indexOf(select_addon) > -1) {
checked = 'checked';
}
// Build our html
html += '<div class="suggested-addon">';
html += '<label class="checkmark-container" for="suggested_attr_addons_' + matched_addons[i].id + '_' + matched_addons[i].addon_name + '">';
html += '<div class="suggested-addon-image">';
html += '<img class="suggested-addons-image lazy" src="' + images_url + 'fetch/w_110,e_sharpen:80,q_auto,f_auto,dpr_auto/' + site_url + 'images/attributes/' + matched_addons[i].image_name + '">';
html += '</div>';
html += '<div class="suggested-addon-name">';
html += matched_addons[i].pretty_name;
html += '</div>';
html += '<div class="addon-price">';
html += '<strong>' + price + '</strong>';
html += '</div>';
html += '<input onChange="checkAddon(\'' + select_addon + '\');" class="suggested-addon-checkmark" type="checkbox" ' + checked + ' ' + readonly + ' toggle-id="suggested_attr_addons_' + matched_addons[i].id + '_' +
new_addon_name + '" id="suggested_attr_addons_' + matched_addons[i].id + '_' + matched_addons[i].addon_name + '" name="suggested_attr_addons_' + matched_addons[i].id + '_' + matched_addons[i].addon_name + '" />';
html += '<span class="checkmark"></span>';
html += '</label>';
html += '</div>';
}
// Add the html to the .suggested-addons and remove the hidden class
$('.suggested-addons').html(html);
$('.occasion-addon-window').removeClass('hidden');
}
}
});
} else {
// No matched occasions so remove the html and add the hidden class
$('.suggested-addons').html('');
$('.occasion-addon-window').addClass('hidden');
}
} else {
// Message is empty so remove the html and add the hidden class
$('.suggested-addons').html('');
$('.occasion-addon-window').addClass('hidden');
}
}
// Check addon function
function checkAddon(id) {
// Get the toggle
var toggle_id = $('input[name="' + id + '"]');
// If the toggle is checked
if (toggle_id.prop('checked') == true) {
toggle_id.prop('checked', false);
} else {
toggle_id.prop('checked', true);
}
}
</script>
<div class="occasion-addon-window hidden">
<div class="occasion-step">
<span class="stage">
<i class="fa-regular fa-gift"></i>
</span>
<span class="product-step-title"> Suggested addons </span>
</div>
<div class="suggested-addons"></div>
</div>
</div>
<script>
$(document).ready(function() {
$('#submit').click(function(e) {
if ($('#personalised').is(':checked') && !$('#attr_personalised').val()) {
e.preventDefault();
window.alert("You haven't personalised your card. Please click Personalise to customise your card, or add a message card instead.");
return false;
}
OpenCrossSell();
});
})
</script>
<script>
var pathArray = window.location.pathname.split('/');
</script>
<!-- Delivery Date -->
<div class="product-step-box step2 mandatory">
<div class="product-step datepicker-trigger">
<span class="stage">
<i class="fa-regular fa-calendar"></i>
</span>
<span class="product-step-title"> Choose a Delivery Date </span>
</div>
<div class="step-content">
<!-- <style>
body.product_page .datepicker{
display: none !important;
}
</style> -->
<div class="form-group list-picker">
<label for="delivery_date">Date</label>
<div class="datepicker-container">
<input id="delivery_date" name="attr_deliverydate" class="form-control datepicker-trigger" value="Thursday, 07th November, 2024">
<img class="datepicker-trigger" alt="Calendar Open" src="https://images.prestigeflowers.co.uk/fetch/https://www.prestigeflowers.co.uk/images/assets/calendar.jpg">
</div>
</div>
<script>
$(function() {
var product_id = '5709';
var input = $('#delivery_date');
var key = 0;
input.on('change', function() {
var delivery_date_selected = input.val();
var datas = delivery_date_selected + '|' + product_id + '|' + key;
psLoad('delivery_option_result', 'delivery_options', datas);
});
psLoad('delivery_option_result', 'delivery_options', input.val() + '|' + product_id + '|' + key);
});
</script>
<!--END DELIVERY DATE-->
<style>
</style>
<!--START PEAK CALENDAR CTA-->
<!--END PEAK CALENDAR CTA-->
<div class="next-3-days">
<span class="next-3-day" data-date="07-11-2024">Thursday, 07, Nov</span>
<span class="next-3-day" data-date="08-11-2024">Friday, 08, Nov</span>
<span class="next-3-day" data-date="09-11-2024">Saturday, 09, Nov</span>
</div>
<div id="delivery_option_result">
<div class="delivery-option premium-delivery paid-delivery">
<div class="radio-button">
<input type="radio" name="attr_deliverymethod_0" id="attr_deliverymethod_0_0" checked="checked" value="Premium Delivery" onchange="SaveCart(0);">
<label for="attr_deliverymethod_0_0">
<strong>Premium Delivery</strong>
<p><strong>Tracked Named Day Delivery (Recommended) - Order by 10pm</strong> <br> Delivered the Next Day. Track your order online & receive SMS updates</p>
<span>£5.99</span>
</label>
</div>
</div>
<div class="delivery-option before-1pm paid-delivery">
<div class="radio-button">
<input type="radio" name="attr_deliverymethod_0" id="attr_deliverymethod_1_0" value="Before 1pm" onchange="SaveCart(0);">
<label for="attr_deliverymethod_1_0">
<strong>Before 1pm</strong>
<p><strong>Guaranteed before 1pm (Recommended) - Order by 2pm</strong> <br> Delivered the Next Day before 1pm. Track your order online & receive SMS updates (requires a signature)</p>
<span>£8.99</span>
</label>
</div>
</div>
<script>
if (typeof clearDelivery !== 'undefined') {
clearDelivery();
}
</script>
</div>
<div class="clearfix mrg20B"></div>
</div>
</div>
<!-- END Delivery Date -->
<div class="step-content">
<div class="afix add-cart-icon" data-spy="affix" data-offset-top="2100">
<input type="hidden" name="csrf_token" value="MTczMDg1MTQyNlBadHpWVVd4NnpNQzE4b0hLdDE0OWhyV1V3Nkd2WlN5"> <input type="hidden" id="productid" name="productid" value="5709">
<input type="hidden" id="attr_premiummsg" name="attr_premiummsg">
<input type="button" name="submit" id="submit" class="add-cart btn" onclick="PinTracking();" value="Order Now »">
<div id="addon_cross_sell_container" active="yes">
<div id="addon_cross_sell_wrapper">
<div class="addons-cross-sell">
<div onclick="CloseCrossSell();" class="close-button">X</div>
<span class="addon_id hidden">1007</span>
<img src="https://images.prestigeflowers.co.uk/fetch/https://www.prestigeflowers.co.uk//images/something-extra/pf-popup-updated-ayr.png" alt="Add a little something extra" class="addon-img">
<hr>
<div class="row">
<div class="col-12 col-md-7">
<h3>Special Offer: Add a Luxury Chocolates 180g +£4.99</h3>
</div>
<div class="col-12 col-md-5">
<div class="row">
<div class="col-12 col-md-6">
<input type="submit" class="btn error" value="No Thanks">
</div>
<div class="col-12 col-md-6">
<input type="submit" class="btn success" onclick="ToggleAddon('attr_addons_1007_180gPralines');" value="Yes Please">
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</form>
POST https://www.prestigeflowers.co.uk/search
<form method="POST" action="https://www.prestigeflowers.co.uk/search">
<select name="search_box" placeholder="All Categories">
<option value="">All Categories</option>
<option value="Anniversary Flowers">Anniversary Flowers</option>
<option value="Birthday Flowers">Birthday Flowers</option>
<option value="Congratulations Flowers">Congratulations Flowers</option>
<option value="Get Well">Get Well</option>
<option value="Just Because">Just Because</option>
<option value="New Baby Flowers">New Baby Flowers</option>
<option value="Next Day Flowers">Next Day Flowers</option>
<option value="Thank You Flowers">Thank You Flowers</option>
<option value="Valentines Day Flowers">Valentines Day Flowers</option>
</select>
<select name="price_range" placeholder="All Prices">
<option value="0">All Prices</option>
<option value="0-30">Below £30</option>
<option value="30-40">£30 to £40</option>
<option value="40-50">£40 to £50</option>
<option value="50-0">Above £50</option>
</select>
<button type="submit" class="btn btn-block">Find</button>
</form>
<form>
<div class="errors">
<label for="nl-name"></label>
</div>
<div class="form-group name">
<input type="text" class="form-control name-box" id="nl-name" aria-describedby="emailHelp" placeholder="Name">
</div>
<div class="errors">
<label for="nl-email"></label>
</div>
<div class="form-group email">
<input type="email" class="form-control email-box" id="nl-email" placeholder="Email Address">
</div>
<!-- <button type="button" class="btn get-code">Subscribe</button> -->
<div class="newsletter-birthday-container">
<p for="nl-date-of-birth" class="birthday-text">Why not add your Birthday, so we know when to send you a free gift!</p>
<div class="newsletter-birthday-fields">
<input type="date" max="2024-11-06" class="form-control birthday-box" id="nl-date-of-birth" placeholder="Date of Birth">
</div>
</div>
<button type="button" class="g-recaptcha btn get-code" data-sitekey="6Lc3w6kZAAAAAJBuVZeCy9sh3HhaeOkW7PiMVHvu" data-badge="inline" data-size="invisible" data-callback="newsletterSubscribe" data-action="submit"
style="color: #ffffff; background: #000000;">Sign Up</button>
<button type="button" class="btn no-thanks" data-dismiss="modal" style="color: #fff !important;">No, Thank You</button>
</form>
Text Content
verifying... Need help? Call us for free on 0344 310 5555 0 Excellent Welcome to Prestige Flowers Sign In or Register My Account Order Tracking Customer Service -------------------------------------------------------------------------------- Your Basket Is Empty * Next Day Flowers NEXT DAY FLOWERS * Next Day Flowers * Birthday Flowers * Autumn Flowers * Flowers with Free Chocs * Florist Specials * Luxury Flowers * Charity Flowers * Back * Birthday Flowers BIRTHDAY FLOWERS * Birthday Flowers * Birthday Gifts * Birthday Flowers With Cards * Birthday Gift Sets * Postal Gifts * Back * By Occasion BY OCCASION * Birthday Flowers * Sympathy Flowers * Anniversary Flowers * Romantic Flowers * Thank You Flowers * Get Well Soon Flowers * Congratulations Flowers * New Baby Flowers * Christmas Flowers * Back * Christmas CHRISTMAS * Christmas Flowers * Christmas Flowers with Card * Luxury Christmas Flowers * Christmas Plants * Letterbox Christmas Flowers * Christmas Flowers with Fizz * Christmas Flowers with Chocolate * Christmas Flowers with Postal Gifts * Christmas Hampers * Christmas Plant Gifts * Back * Luxury LUXURY * Luxury Flowers * Haute Florist © * Hat Boxes * Flowers & Champagne * Castle Howard Collection * National Gallery Flowers * Back * Letterbox Flowers LETTERBOX FLOWERS * All Letterbox Flowers * Letterbox Subscription * Back * Plants PLANTS * Plants * Christmas Plants * Autumn Plants * Indoor Plants * Plant Gifts * Artificial Plants * Outdoor Plants * Back * Gifts GIFTS * Autumn Gifts * Bouquets & Gift Sets * Flowers with Vase * Flowers with Champagne * Flowers with Chocolates * Postal Gifts * Plant Gifts * Christmas Hampers * Back * Christmas Hampers CHRISTMAS HAMPERS * Christmas Hampers * Christmas Family Gifts * Luxury Christmas Hampers * Christmas Chocolate Gifts * Non Alcoholic Hampers * Christmas Gifts Under £50 * Christmas Eve Boxes * Secret Santa * Back * Flowers with Card FLOWERS WITH CARD * Flowers with Card * Birthday Flowers with Card * Back * Subscriptions * * Login * Register * Track Your Order * Customer Service * Close Menu International Flowers ▼ * Australia * Belgium * Canada * Denmark * France * Germany * Holland * Republic of Ireland * Italy * Japan * Luxembourg * Philippines * Singapore * Spain * United States * United Kingdom * All Countries ORDER BY 10PM FOR NEXT DAY DELIVERY Loading... Delivery from Thursday Order Now! Product Code: CRF005 * MAJESTIC £50 Description Substitution Delivery Info A glorious selection of luxurious red Avalanche Roses arranged along with golden orbits and foliage for a superb floral masterpiece. All professionally arranged in a stunning Prestige Luxury Collection bag, this bunch will impress anyone lucky enough to be on the receiving end. 25% of the value of this gift will be donated to Cancer Research UK to fund research for beating cancer. Due to seasonal availability of certain flowers and colours from time to time, we may use a substitution. In cases like this, we will always use the best match with the stem pictured, and it will always be of equal or higher value to the one pictured. In the rare case that no suitable substitution exists we will make contact with you to find an alternative solution. This product is delivered by trackable courier. Kindly note that some non-mainland areas might require an additional working day for transit. Adverse weather conditions can also result in delays. Deliveries might be early during particular events. For more details, feel free to contact us, and one of our customer care agents will provide you with further details before you place your order. YOU MIGHT ALSO LIKE THESE Lily Rose Medley £27 Add to Basket Order Now » Next Day Delivery Save £25 Rose and Lily £49.99 £24.99 Add to Basket Order Now » Next Day Delivery Save £25 Amaryllis Allure £59.99 £34.99 Add to Basket Order Now » Next Day Delivery Add a Little Extra? Premium Feed FREE! Premium Feed Luxury Chocolates 180g £4.99 Luxury Chocolates 180g Deluxe Chocolates 400g £9.99 Deluxe Chocolates 400g Clear Vase £4.99 Clear Vase Afternoon Tea Hamper £16.99 Afternoon Tea Hamper Guylian Chocolates £2.99 Guylian Chocolates Birthday Cake £4.99 Birthday Cake Luxury Teddy Bear £9.99 Luxury Teddy Bear Premium Belgian Chocolates £19.99 Premium Belgian Chocolates Just For You Chocolate Bar £4.99 Tea and Biscuits £3.99 Sparkling French Wine (75cl) £19.99 Happy Birthday Gift Box £18.99 Birthday Chocolate Bar £4.99 Garden Blooms Candle £9.99 Afternoon Tea Gardening Gift £16.99 Daffodil Bulbs £4.99 Wine Hamper £16.99 Champagne 75cl £34.99 Gin + Tonic Candle £5.99 White Wine (37.5cl) £7.99 Eco Clear Vase £8.99 Dark Chocolate & Raspberry Bar £4.99 Happy Birthday Pick £2.99 Red Wine (37.5cl) £7.99 Happy Birthday Candle £2.99 New Baby Gift £24.99 Perfect Pamper Gift £16.99 Hand Blown Vase £9.99 Fizzy Lemon Sweets £3.99 Chocolate Gift Box Gift £16.99 Moet 75cl £65 Occasion Candle £4.99 Tulip Planting Bulbs £4.99 Moet (37.5cl) £39.99 Gin 5cl + Tonic £4.99 Happy Birthday Afternoon Tea £16.99 Gin and Tonic Gift £16.99 Write a Message Card Add a Personalised Card * Luxury hand finished cards * Cards for all occasions * Choose from many designs * Easily upload your pictures + £3.99 £5.99 Send a FREE recorded message Record your message We'll send with your gift Record Now! SEND YOUR FREE VIDEO MESSAGE WITH EVERY GIFT RECORD A VOICE MESSAGE Just press the 'Record' button to start recording your voice message, you will have up to 30 seconds. PLEASE ALLOW ACCESS TO YOUR CAMERA AND MICROPHONE TO START RECORDING. PLEASE ALLOW ACCESS TO YOUR MICROPHONE TO START RECORDING. LET'S GET STARTED HAPPY WITH YOUR RECORDING? Back Add To Order! Save & Review Record Video Message Voice Message Close Back Record Save & Send Stop Recording Your recording has been saved and a copy will be sent with this product. -------------------------------------------------------------------------------- Or -------------------------------------------------------------------------------- Message Card Standard Message Card 200 Characters Remaining Suggested addons Choose a Delivery Date Date Thursday, 07, Nov Friday, 08, Nov Saturday, 09, Nov Premium Delivery Tracked Named Day Delivery (Recommended) - Order by 10pm Delivered the Next Day. Track your order online & receive SMS updates £5.99 Before 1pm Guaranteed before 1pm (Recommended) - Order by 2pm Delivered the Next Day before 1pm. Track your order online & receive SMS updates (requires a signature) £8.99 X 1007 -------------------------------------------------------------------------------- SPECIAL OFFER: ADD A LUXURY CHOCOLATES 180G +£4.99 Description Substitution Delivery Info A glorious selection of luxurious red Avalanche Roses arranged along with golden orbits and foliage for a superb floral masterpiece. All professionally arranged in a stunning Prestige Luxury Collection bag, this bunch will impress anyone lucky enough to be on the receiving end. 25% of the value of this gift will be donated to Cancer Research UK to fund research for beating cancer. Due to seasonal availability of certain flowers and colours from time to time, we may use a substitution. In cases like this, we will always use the best match with the stem pictured, and it will always be of equal or higher value to the one pictured. In the rare case that no suitable substitution exists we will make contact with you to find an alternative solution. This product is delivered by trackable courier. Kindly note that some non-mainland areas might require an additional working day for transit. Adverse weather conditions can also result in delays. Deliveries might be early during particular events. For more details, feel free to contact us, and one of our customer care agents will provide you with further details before you place your order. YOU MIGHT ALSO LIKE THESE Breast Cancer Awareness Bouquet £26 Add to Basket Order Now » Next Day Delivery Lily Rose Medley £27 Add to Basket Order Now » Next Day Delivery Special Birthday £40 Add to Basket Order Now » Next Day Delivery Reviews Product Finder All Categories Anniversary Flowers Birthday Flowers Congratulations Flowers Get Well Just Because New Baby Flowers Next Day Flowers Thank You Flowers Valentines Day Flowers All Prices Below £30 £30 to £40 £40 to £50 Above £50 Find Copyright © 2024 Prestige Flowers™ Follow Us On: Get our special offers * About Us * Contact Us * Terms & Conditions * Privacy Policy * 100% Satisfaction Guarantee * Delivery Information * Business Opportunities * Flower Care * Plant Care * Student Discount Proud to be sustainable and one of the UK's most ethical florists - See The Stats * FLOWER DELIVERY -------------------------------------------------------------------------------- * Mother's Day Flowers * Valentine's Day Flowers * Christmas Flowers * International Mother's Day * NEXT DAY FLOWERS -------------------------------------------------------------------------------- * Next Day Flowers * Anniversary Flowers * Get Well Soon Flowers * Things You Never Knew About Flowers * 100% SATISFACTION GUARANTEE -------------------------------------------------------------------------------- * Cut flowers – which ones last the longest Thu, 04 Jan 2024 * Flower buds and how to help them open Mon, 16 Oct 2023 -------------------------------------------------------------------------------- Cookies? We are not the cookie monster, we use cookies to allow us to provide the best experience to our customers, by browsing our site you are agreeing to accept these cookies. ✕ SIGN UP AND GET 10% OFF YOUR FIRST ORDER Simply add your name and email below to recieve exclusive offers and news, as well as 10% off your first order! Why not add your Birthday, so we know when to send you a free gift! Sign Up No, Thank You Click here to view our Privacy Policy This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply. By using our services you agree to our use of cookies to improve your shopping experience. View Privacy Policy Accept Privacy Policy SEARCH FOR PRODUCTS We will show you all available products which are related to your query. Chat with us, powered by LiveChat