We’re blessed to know that our resources encouraged your family.
\n
We want you to have this inspirational new devotional from bestselling author Jackie Hill Perry, Upon Waking. It’s an invitation to step away from your to-do lists and anxiety and into a daily reminder of what truly matters.
\n
It is our gift to you when you become one of the X new monthly donors we\'re looking for today. As a FamilyLife partner, your gift will strengthen your family and other families all year. \n
',document.body.appendChild(o),a.querySelector("svg").onclick=()=>n(a),o.querySelector("svg").onclick=()=>n(o),a.querySelector("#popup-no").onclick=()=>{const e=Date.now()+6048e5;localStorage.setItem("popupDismissedUntil",String(e)),n(a)},a.querySelector("#popup-yes").onclick=()=>{n(a),setTimeout(l,300)},setTimeout(()=>a.showModal(),3e3)}();}catch(e) {VWO._.vAEH(e);}
return vwo_$('head')[0] && vwo_$('head')[0].lastChild;})("head")}}, R_725969_176_1_3_0:{ fn:function(log,nonce=''){return (function(x) {
try{
var ctx=vwo_$(x),el;
/*vwo_debug log("Revert","content",""); vwo_debug*/;
el=vwo_$('[vwo-element-id="1746468826149"]');
el.revertContentOp().remove();
} catch(e) {VWO._.vAEH(e);}
try{
var el,ctx=vwo_$(x);
/*vwo_debug log("Revert","addElement","body"); vwo_debug*/(el=vwo_$('[vwo-element-id="1746468826150"]')).remove();
} catch(e) {VWO._.vAEH(e);}
return vwo_$('head')[0] && vwo_$('head')[0].lastChild;})("head")}}, C_725969_176_1_3_1:{ fn:function(log,nonce=''){return (function(x) {var el,ctx=vwo_$(x);
/*vwo_debug log("_clickElement",".btn"); vwo_debug*/(el=vwo_$(".btn")).addClass("_vwo_coal_1697827292277");})(".btn")}}, R_725969_176_1_3_1:{ fn:function(log,nonce=''){return (function(x) {
if(!vwo_$.fn.vwoRevertHtml){
return;
};
var el,ctx=vwo_$(x);
/*vwo_debug log("Revert","_clickElement",".btn"); vwo_debug*/(el=vwo_$(".btn")).removeClass("_vwo_coal_1697827292277");})(".btn")}}, C_725969_183_1_2_0:{ fn:function(log,nonce=''){return (function(x) {
try{
var _vwo_sel = vwo_$("`);
!vwo_$("head").find('#1746479885395').length && vwo_$('head').append(_vwo_sel);}catch(e) {VWO._.vAEH(e);}
try{}catch(e) {VWO._.vAEH(e);}
try{function getCurrentDate (d = new Date()) { return d.toISOString().split('T')[0] }
function vwoCustomEvent (labelValue) {
window.VWO = window.VWO || [];
VWO.event = VWO.event || function () {VWO.push(["event"].concat([].slice.call(arguments)))};
VWO.event("customEvent", { "label": labelValue.toString() });
}
function vwoDonationInterrupter2 (id, type) {
window.VWO = window.VWO || [];
VWO.event = VWO.event || function () {VWO.push(["event"].concat([].slice.call(arguments)))};
if (type === "shown" || type === "error") {
VWO.event("donationInterrupter2", {
"id": id?.toString(),
"type": type?.toString(),
});
} else {
VWO.event("donationInterrupter2action", {
"id": id?.toString(),
"type": type?.toString(),
});
}
}
class SortOfSimpleCountdownTimer{constructor(t,i=document.createElement("div"),e={onEnd:function(t){t.hide()},onChange:function(t){}}){return console.log(t),this._time={end:t instanceof Date?t:new Date(t).getTime(),now:new Date().getTime()},this._time.remaining=this._time.end-this._time.now,this._data=null,i.classList.contains("countdown")||i.classList.add("countdown"),this.element=i,this.callbacks=e,setTimeout(()=>this.init(),50),this}init(){console.log("Countdown Timer initialized."),this._interval=setInterval(t=>t.update(),1e3,this),this.update()}update(){try{this.now=new Date().getTime(),this.remaining=this.end-this.now,this.now;let t=this.remaining;if(!this.remaining||"NaN"==this.remaining){console.warn("Remaining time is not a number or is falsy.");return}if(this.remaining<=0)this.finish(),this.hide();else{let i={days:Math.floor(t/864e5),hours:Math.floor(t%864e5/36e5),minutes:Math.floor(t%36e5/6e4),seconds:Math.floor(t%6e4/1e3)};this.data=i;let e=[];Object.entries(i).filter(([t,i])=>"seconds"==t).forEach(([t,i])=>e.push(`
${i||0}
`)),this.element.innerHTML=e.join("\n")}this.callbacks&&"function"==typeof this.callbacks.onChange&&this.callbacks.onChange.call(this,this.element)}catch(n){this.element.style.setProperty("display","none"),console.error(n)}}finish(){this._interval?(clearInterval(this._interval),console.info("Countdown finished.")):console.error("Failed to clear interval."),this.callbacks&&"function"==typeof this.callbacks.onEnd&&this.callbacks.onEnd.call(this,this.element)}hide(){this.element.style.setProperty("display","none")}get now(){return this._time.now}set now(t){this._time.now=t||new Date().getTime()}get end(){return this._time.end}set end(t){this._time.end=t}get remaining(){return this._time.remaining}set remaining(t){this._time.remaining=t||this._time.end-this._time.now}get data(){return this._data}set data(t){this._data=t}}
class NextAfterDonationMicrosite_GiftArrayButton {
static activeClass = 'active';
constructor (element) {
if (!(element instanceof HTMLElement) || !(element.classList.contains('amt-button') || element.classList.contains('option')))
throw new TypeError("Argument 0 must be an instance of HTMLElement with a class of '.amt-button' or '.option'.");
this.button = element;
}
get value () {
return parseInt(this.button.dataset.amt);
}
set value (newValue) {
this.button.dataset.amt = parseInt(newValue);
}
get text () {
return this.button.textContent;
}
set text (newText) {
this.button.textContent = newText;
}
get active () {
return this.button.classList.contains(NextAfterDonationMicrosite_GiftArrayButton.activeClass) ? true : false;
}
set active (boolean) {
if (boolean) {
this.select();
if (!this.active) // if for some reason the button was selected but still does not have the active class
this.button.classList.add(NextAfterDonationMicrosite_GiftArrayButton.activeClass);
} else {
this.button.classList.remove(NextAfterDonationMicrosite_GiftArrayButton.activeClass);
}
}
click () { this.button.click() }
select () { this.click() }
addEventListener (eventType, eventHandlerCallback) {
this.button.addEventListener(eventType, eventHandlerCallback);
}
get amount () {
return this.value;
}
set amount (newAmount) {
if (Number.isNaN(parseInt(newAmount)) || parseInt(newAmount) < 0)
throw new Error("New amount must be a valid number greater than 0.");
newAmount = parseInt(newAmount);
this.text = parseFloat(newAmount).toLocaleString('en-US', { format: 'currency', currency: 'USD', minimumFractionDigits: 0, maximumFractionDigits: 0 });
this.value = parseInt(newAmount);
}
}
class NextAfterDonationMicrosite_GiftArrayButtonRadio extends NextAfterDonationMicrosite_GiftArrayButton {
constructor (element) {
super(element);
this.radio = element.matches('input[type="radio"]') ? element : element.querySelector('input[type="radio"]');
}
get value () {
return parseInt(this.radio.value);
}
set value (newValue) {
this.radio.value = parseInt(newValue);
}
set text (newText) {
const textNode = this.radio.nextSibling.nodeType === Node.TEXT_NODE ? this.radio.nextSibling : [...this.element.childNodes].find(({ nodeType, nodeValue }) => nodeType === Node.TEXT_NODE && nodeValue.length > 0);
textNode.nodeValue = newText.toString();
}
addEventListener (eventType, eventHandlerCallback) {
if (eventType == 'click') {
this.button.addEventListener(eventType, eventHandlerCallback);
} else {
this.radio.addEventListener(eventType, eventHandlerCallback);
}
}
}
class NextAfterDonationMicrosite_GiftArrayOtherAmount {
constructor (element) {
if (!(element instanceof HTMLInputElement) || !(element.type == 'text'))
throw new TypeError("Argument 0 must be an instance of HTMLInputElement with a 'type' of 'text'.");
this.input = element;
}
get value () {
return this.input.value;
}
set value (newValue) {
this.input.dispatchEvent(new Event('click'));
this.input.dispatchEvent(new Event('focus'));
this.input.value = newValue;
// this.input.dispatchEvent(new Event('keyup'));
this.input.dispatchEvent(new Event('input'));
this.input.dispatchEvent(new Event('change'));
this.input.dispatchEvent(new Event('blur'));
}
get text () {
return this.input.placeholder;
}
set text (newPlaceholderText) {
this.input.placeholder = newPlaceholderText;
}
get amount () {
return parseFloat(this.value);
}
set amount (newAmount) {
if (Number.isNaN(parseInt(newAmount)) || parseInt(newAmount) <= 0)
throw new Error("New amount must be a valid number greater than 0.");
newAmount = parseFloat(newAmount);
this.value = newAmount;
}
addEventListener (eventType, eventHandlerCallback) {
this.input.addEventListener(eventType, eventHandlerCallback);
}
}
class NextAfterDonationMicrosite_GiftArray extends Array {
constructor (items) {
if (!Array.isArray(items) && items.length === 0) {
throw !Array.isArray(items) ? new TypeError("GiftArray: Arugment 1 is not an instance of Array:" + items.join(', ')) : new RangeError("GiftArray: Arugment 1 is not an instance of Array with a length greater than 0:" + items.join(', '));
}
if (items.every((item) => item instanceof NextAfterDonationMicrosite_GiftArrayButton || item instanceof NextAfterDonationMicrosite_GiftArrayOtherAmount)) {
if (items.find((item) => item instanceof NextAfterDonationMicrosite_GiftArrayOtherAmount)) { // if the array is mixed (both GiftArrayButton and GiftArrayButtonOtherAmount)
let temp = items.find((item) => item instanceof NextAfterDonationMicrosite_GiftArrayOtherAmount); // temporarily hold the other amount
items = items.filter((item) => item instanceof NextAfterDonationMicrosite_GiftArrayButton); // assign items to be only the gift array button temporarily (removed the other amount)
items.push(temp); // add back the other amount to items so that it's always the last entry in the array
}
} else if (items.every((item) => item instanceof HTMLElement)) {
items = items.map((item) => {
if (item.matches('.stripe-donation-other-amt, .option-other-text'))
return new NextAfterDonationMicrosite_GiftArrayOtherAmount(item);
return item.querySelector('input[type="radio"]') ? new NextAfterDonationMicrosite_GiftArrayButtonRadio(item) : new NextAfterDonationMicrosite_GiftArrayButton(item);
});
} else {
throw new Error("GiftArray: Arugment 1 is not of type HTMLElement[], or GiftArrayButton[]:" + items.join(', '));
}
super(...items);
this.Buttons = items.filter((item) => item instanceof NextAfterDonationMicrosite_GiftArrayButton) ?? [];
this.OtherAmountInput = items.find((item) => item instanceof NextAfterDonationMicrosite_GiftArrayOtherAmount) ?? null;
}
get amount () {
const activeButton = this.Buttons.find((item) => item.active);
if (activeButton) { // active button was found in the gift array
return activeButton.amount;
} else {
return this.OtherAmountInput.amount;
}
}
set amount (newAmount) {
if (Number.isNaN(parseInt(newAmount)) || parseInt(newAmount) <= 0)
throw new Error("New amount must be a valid number greater than 0.");
newAmount = parseFloat(newAmount);
const matchingButton = this.find((item) => parseFloat(item.value) === newAmount);
if (matchingButton) {
matchingButton.click();
} else {
this.OtherAmountInput.amount = newAmount;
const activeButton = this.Buttons.find((item) => item.active);
if (activeButton) activeButton.active = false;
}
}
addEventListeners (eventType, eventHandlerCallback) {
this.forEach((item) => item.addEventListener(eventType, eventHandlerCallback));
}
}
class NextAfterDonationMicrosite_FrequencyToggle {
constructor (element) {
if (!(element instanceof HTMLInputElement) || !(element.type == 'checkbox'))
throw new TypeError("Argument 0 must be an instance of HTMLInputElement with a 'type' of 'checkbox'.");
this.checkbox = element;
}
get frequency () {
return this.checkbox.checked ? true : false;
}
set frequency (boolean) {
if (boolean !== this.frequency) { // only if the arugment is different from what is already set
if (boolean === true)
this.checkbox.checked = true;
if (boolean === false)
this.checkbox.checked = true;
}
}
get recurring () {
return this.frequency ? true : false;
}
set recurring (bool) {
this.frequency = bool;
}
addEventListener (eventType, eventHandlerCallback) {
this.checkbox.addEventListener(eventType, eventHandlerCallback);
}
}
class NextAfterDonationMicrosite_FrequencyRadio {
constructor (element) {
if (!(element instanceof HTMLInputElement) || !(element.type == 'radio'))
throw new TypeError("Argument 0 must be an instance of HTMLInputElement with a 'type' of 'radio'.");
this.radio = element;
}
get value () {
return this.radio.value;
}
set value (newValue) {
this.radio.value = newValue;
}
get frequency () {
return this.radio.checked ? true : false;
}
set frequency (boolean) {
if (boolean !== this.frequency) { // only if the arugment is different from what is already set
if (boolean === true)
this.radio.checked = true;
if (boolean === false)
this.radio.checked = true;
}
}
get recurring () {
return this.frequency ? true : false;
}
set recurring (bool) {
this.frequency = bool;
}
get checked () {
return this.radio.checked;
}
set checked (bool) {
this.radio.checked = bool ? true : false;
}
click () {
this.radio.click();
}
addEventListener (eventType, eventHandlerCallback) {
this.radio.addEventListener(eventType, eventHandlerCallback);
}
}
class NextAfterDonationMicrosite_FrequencyArray extends Array {
constructor (items) {
if (!Array.isArray(items) && items.length === 0) {
throw !Array.isArray(items) ? new TypeError("FrequencyArray: Arugment 1 is not an instance of Array:" + items.join(', ')) : new RangeError("FrequencyArray: Arugment 1 is not an instance of Array with a length greater than 0:" + items.join(', '));
}
if (items.every((item) => item instanceof NextAfterDonationMicrosite_FrequencyRadio)) {
items = items;
} else if (items.every((item) => item instanceof HTMLElement)) {
items = items.map((item) => (item instanceof HTMLInputElement && item.type === 'radio') ? new NextAfterDonationMicrosite_FrequencyRadio(item) : undefined);
} else {
throw new Error("FrequencyArray: Arugment 1 is not of type HTMLInputElement[]|HTMLElement[], or FrequencyRadio[]:" + items.join(', '));
}
super(...items);
this.Options = items.filter((item) => item instanceof NextAfterDonationMicrosite_FrequencyRadio) ?? [];
}
get frequency () {
const checkedRadio = this.Options.find(({ radio }) => radio.checked);
if (checkedRadio) { // active button was found in the gift array
return checkedRadio.value;
} else {
return undefined;
}
}
set frequency (newFrequency) {
const matchingRadio = this.Options.find(({ value }) => value === newFrequency);
if (matchingRadio) {
matchingRadio.click();
} else {
console.warn("Invalid frequency: ", newFrequency);
}
}
addEventListener (eventType, eventHandlerCallback) {
this.Options.forEach((option) => option.addEventListener(eventType, eventHandlerCallback));
}
}
class NextAfterDonationMicrosite_FrequencyTab {
constructor (element) {
if (!(element instanceof HTMLElement))
throw new TypeError("Argument 0 must be an instance of HTMLElement.");
this.element = element;
}
get text () {
return this.element.textContent;
}
set text (newText) {
this.element.textContent = newText;
}
click () {
this.element.click();
}
addEventListener (eventType, eventHandlerCallback) {
this.radio.addEventListener(eventType, eventHandlerCallback);
}
}
class NextAfterDonationMicrosite_FrequencyTabs extends Array {
constructor (items) {
if (!Array.isArray(items) && items.length === 0) {
throw !Array.isArray(items) ? new TypeError("FrequencyTabs: Arugment 1 is not an instance of Array:" + items.join(', ')) : new RangeError("FrequencyTabs: Arugment 1 is not an instance of Array with a length greater than 0:" + items.join(', '));
}
if (items.every((item) => item instanceof NextAfterDonationMicrosite_FrequencyTab)) {
items = items;
} else if (items.every((item) => item instanceof HTMLElement)) {
items = items.map((item) => new NextAfterDonationMicrosite_FrequencyTab(item) ?? undefined);
} else {
throw new Error("FrequencyTabs: Arugment 1 is not of type HTMLElement[], or FrequencyTab[]:" + items.join(', '));
}
super(...items);
this.Buttons = items.filter((item) => item instanceof NextAfterDonationMicrosite_FrequencyTab) ?? [];
}
select (value) {
const matchingTab = this.Buttons.find(({ text }) => text.match(value) || text.includes(value));
if (matchingTab) {
matchingTab.click();
} else {
console.warn("Invalid value: ", value);
}
}
addEventListener (eventType, eventHandlerCallback) {
this.Buttons.forEach((option) => option.addEventListener(eventType, eventHandlerCallback));
}
}
//
const lockedProperty = { writable: false, configurable: false, enumerable: true };
function DonationFormAPI (elements, options = {}) {
const defaultOptions = {
min: 1.00,
max: 999999.99,
makeTabbed: false,
fakeSubmit: true,
overrideGiftArrayValues: false,
};
options = { ...defaultOptions, ...options };
//
const { recurringOptions, amountButtons, submitButton, root } = elements;
const debug = {
log: (...args) => window.NA.DonationForm.DEBUG_MODE && console.log(...args),
info: (...args) => window.NA.DonationForm.DEBUG_MODE && console.log(...args),
warn: (...args) => window.NA.DonationForm.DEBUG_MODE && console.log(...args),
error: (...args) => window.NA.DonationForm.DEBUG_MODE && console.log(...args),
};
//
const api = new Object();
Object.defineProperty(api, 'root', {
value: root,
writable: false,
configurable: true,
enumerable: true,
});
Object.defineProperties(api, {
FORM_MINIMUM: { // constant
value: options.min || 0,
...lockedProperty
},
FORM_MAXIMUM: { // constant
value: options.max || Infinity,
...lockedProperty
},
});
Object.defineProperties(api, {
GiftArray: { // variable
value: new NextAfterDonationMicrosite_GiftArray(amountButtons),
writable: false,
configurable: true,
enumerable: true,
},
Frequency: { // variable
value: Array.isArray(recurringOptions) ? new NextAfterDonationMicrosite_FrequencyArray(recurringOptions) : new NextAfterDonationMicrosite_FrequencyToggle(recurringOptions),
writable: false,
configurable: true,
enumerable: true,
},
SubmitButton: { // variable
value: submitButton,
writable: false,
configurable: false,
enumerable: true,
}
});
if (elements.tabs && Array.isArray(elements.tabs) && elements.tabs.length >= 2 && elements.tabs.every((item) => item instanceof HTMLElement)) {
Object.defineProperty(api, 'Tabs', {
value: new NextAfterDonationMicrosite_FrequencyTabs(elements.tabs),
writable: false,
configurable: true,
enumerable: true,
});
}
Object.defineProperties(api, {
getFrequency: { // function
value: async function () {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
/*if (this.Tabs && this.Tabs instanceof NextAfterDonationMicrosite_FrequencyTabs) { // tabs
}
else*/ if (this.Frequency instanceof NextAfterDonationMicrosite_FrequencyArray) { // radios
return new Promise((resolve, reject) => {
try {
resolve(this.Frequency.frequency);
} catch (error) {
reject(error);
}
});
}
else if (this.Frequency instanceof NextAfterDonationMicrosite_FrequencyToggle) { // checkbox
return new Promise((resolve, reject) => {
try {
resolve(this.Frequency.frequency);
} catch (error) {
reject(error);
}
});
}
}, ...lockedProperty
},
setFrequency: { // function
value: async function (frequency) {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
if (this.Frequency instanceof NextAfterDonationMicrosite_FrequencyArray) { // radios
if (this.Tabs && this.Tabs instanceof NextAfterDonationMicrosite_FrequencyTabs) { // tabs
let index = this.Frequency.Options.findIndex(({ value }) => value.trim().match(new RegExp(frequency, 'i')) || value.trim().includes(frequency)) ?? -1;
if (index > -1 && index < this.Tabs.Buttons.length) {
this.Tabs.Buttons[index]?.click();
} else {
this.Tabs.select(frequency);
}
}
return new Promise(async (resolve, reject) => {
try {
this.Frequency.frequency = frequency;
if (await this.getFrequency() === frequency)
resolve(frequency);
} catch (error) {
reject(error);
}
});
}
if (this.Frequency instanceof NextAfterDonationMicrosite_FrequencyToggle) { // checkbox
if (this.Tabs && this.Tabs instanceof NextAfterDonationMicrosite_FrequencyTabs) { // tabs
let index = this.Frequency.Options.findIndex(({ value }) => value.trim().match(new RegExp(frequency, 'i')) || value.trim().includes(frequency)) ?? -1;
if (index > -1 && index < this.Tabs.Buttons.length) {
this.Tabs.Buttons[index]?.click();
} else {
this.Tabs.select(frequency);
}
}
return new Promise(async (resolve, reject) => {
try {
this.Frequency.frequency = frequency;
if (await this.getFrequency() === frequency)
resolve(frequency);
} catch (error) {
reject(error);
}
});
}
}, ...lockedProperty
},
getAmount: { // function
value: async function () {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
return new Promise((resolve, reject) => {
try {
resolve(this.GiftArray.amount);
} catch (error) {
reject(error);
}
});
}, ...lockedProperty
},
setAmount: { // function
value: async function (amount, frequency = undefined) {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
return new Promise(async (resolve, reject) => {
try {
this.GiftArray.amount = amount;
if (await this.getAmount() === amount)
resolve(amount);
} catch (error) {
reject(error);
}
});
}, ...lockedProperty
},
getRecurring: { // function
value: async function () {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
return new Promise((resolve, reject) => {
try {
resolve(this.Frequency.recurring);
} catch (error) {
reject(error);
}
});
}, ...lockedProperty
},
setRecurring: { // function
value: async function (bool) {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
return new Promise(async (resolve, reject) => {
try {
this.Frequency.frequency = bool ? true : false;
if (await this.getRecurring() === bool)
resolve(bool);
} catch (error) {
reject(error);
}
});
}, ...lockedProperty
},
freqency: {
get () { return this.getFrequency() },
set (value) { this.setFrequency(value) },
enumerable: true, configurable: true,
},
amount: {
get () { return this.getAmount() },
set (value) { this.setAmount(value) },
enumerable: true, configurable: true,
},
recurring: {
get () { return this.getRecurring() },
set (value) { this.setRecurring(value) },
enumerable: true, configurable: true,
},
});
Object.defineProperties(api, {
submit: { // function
value: async function (condition = this.validate||function(){return true}) {
//this.hooks['onBeforeSubmit'].forEach((callback) => callback.call(this));
let result;
const isAsyncFunction = (func) => func.constructor.name === "AsyncFunction";
if (Array.isArray(condition)) {
if (condition.every((c) => typeof c === 'function' && isAsyncFunction(c))) {
result = await Promise.all(condition.map(async (c) => await c.call(this)));
} else if (condition.every((c) => typeof c === 'function')) {
result = condition.every((c) => c.call(this));
} else if (condition.every((c) => c === true || c === false)) {
result = condition.every((c) => c);
}
} else if (typeof condition === 'function' && isAsyncFunction(condition)) {
result = await condition.call(this);
} else if (typeof condition === 'function') {
result = condition.call(this);
} else if (condition === true || condition === false) {
result = condition;
} else {
console.error("Unknown error.");
debugger;
}
//
if (result === true) {
if (window.NA.DonationForm.hasOwnProperty("DEBUG_MODE") && window.NA.DonationForm["DEBUG_MODE"] == true)
return console.log("Submit aborted (debug mode is enabled).");
this.SubmitButton.click(),
this.hooks['onSubmit'].forEach((callback) => callback.call(this));
//this.hooks['onAfterSubmit'].forEach((callback) => callback.call(this));
} else {
return console.log("Submit failed (conditions did not evaluate to true).");
}
}, ...lockedProperty
},
interceptSubmit: { // function
value: function (handleInterceptedSubmit = () => { return new Promise((resolve) => resolve(undefined)) }) {
try {
window.NA.DonationForm.SubmitButtonCopy = window.NA.DonationForm.SubmitButtonCopy || createNewSubmitButton(window.NA.DonationForm.SubmitButton, { cloneOriginal: false, hideOriginal: true, observeOriginal: false });
window.NA.DonationForm.SubmitButtonCopy.addEventListener('click', async (event) => {
event.preventDefault(), event.stopPropagation();
const shouldRetryInterrupterAfterFailedSubmit = false;
const shouldFormSubmit = await handleInterceptedSubmit.call(this, event);
if (shouldFormSubmit) {
console.info("Submit allowed by initial interceptSubmit callback function resulting in a truthy evaluation.");
const formIsValid = await window.NA.DonationForm.validate();
if (!formIsValid) { // if the the form SHOULD submit (based on handleInterceptedSubmit return value) and if the form IS NOT valid (there are known errors in the form found by calling the form API's validate function)
console.warn("Form has known errors. Attempting to submit to show errors then retrying.");
console.log("Submitting...");
window.NA.DonationForm.submit(true); // submit anyway to trigger the error to be shown
if (shouldRetryInterrupterAfterFailedSubmit) { // if true, the fake button that will intercept submit will be shown again after a failed submit, otherwise it will show the original to ensure it submits the next time
window.NA.DonationForm.SubmitButton.style.setProperty("display", "none"), debug.info("SubmitButton hidden."); // hide the original submit button again
// window.NA.DonationForm.SubmitButtonCopy.style.setProperty("display", "none"), debug.info("SubmitButtonCopy hidden."); // hide the copy of the submit button again
window.NA.DonationForm.SubmitButtonCopy.style.removeProperty("display"), debug.info("SubmitButtonCopy unhidden."); // show the copy of the submit button
} else {
window.NA.DonationForm.SubmitButtonCopy.style.setProperty("display", "none"), debug.info("SubmitButtonCopy hidden."); // hide the copy of the submit button again
// window.NA.DonationForm.SubmitButton.style.setProperty("display", "none"), debug.info("SubmitButton hidden."); // hide the copy of the submit button again
window.NA.DonationForm.SubmitButton.style.removeProperty("display"), debug.info("SubmitButton unhidden."); // hide the original submit button again
}
} else { // if the the form SHOULD submit (based on handleInterceptedSubmit return value) and if the form IS valid
console.log("Submitting...");
window.NA.DonationForm.submit(true);
}
} else { // if the form SHOULD NOT submit (based on handleInterceptedSubmit return value) -- usually means show donation interrupter after preventing submit action
console.log("Submit prevented."), console.info("Next submit will be allowed.");
window.NA.DonationForm.SubmitButton.style.removeProperty("display"), debug.info("SubmitButton unhidden."); // show the original submit button so that if something goes wrong the user can still click the submit button
window.NA.DonationForm.SubmitButtonCopy.style.setProperty("display", "none"), debug.info("SubmitButtonCopy hidden."); // hide the copy of the submit button that intercepts submit attempts so that there aren't two buttons
}
});
console.log("Submit intercept added.\nButton:", window.NA.DonationForm.SubmitButtonCopy);
} catch (error) {
console.error("Failed to add submit intercept:", error);
}
}, ...lockedProperty
},
validate: { // function
value: async function (root = undefined) {
if (!this || this === null)
throw new Error("validate: Unable to read API context.");
root = root || this.root;
const flattenArray = (array) => array.reduce((flat, toFlatten) => flat.concat(Array.isArray(toFlatten) ? flattenArray(toFlatten) : toFlatten), []);
try {
const isRecurring = await this.getFrequency(),
amount = await this.getAmount();
if (isRecurring === null || isRecurring === undefined)
return false;
if (!amount)
return false;
if (amount < this.FORM_MINIMUM || amount > this.FORM_MAXIMUM)
return console.error("validate:", "Gift amount is invalid:", amount), false;
const chosenPaymentMethod = root.querySelector('.payment-option-holder .active')?.textContent.trim();
const requiredFields = Array.from(root.querySelectorAll('input[required]'))
const valid = requiredFields.every((input) => {
const type = input.tagName.toLowerCase() === 'select' ? 'select' : input.type;
const { name, value, id } = input;
//console.log(type, name, value);
if (input.classList.contains('stripe-donation-card-number')) { // if the required input is the CC field AND the selected payment is CC
if (chosenPaymentMethod == 'Credit Card') {
if (input.value.trim().length === 16) // if the card number is 16 digits
return true; // mark as valid
return false;
} else { // CC isn't even selected so don't even need to validate it
return true;
}
}
if (input.classList.contains('stripe-donation-routing-number') || input.classList.contains('stripe-donation-account-number') || input.classList.contains('stripe-donation-account-number-check')) { // if the required input is the CC field AND the selected payment is CC
if (chosenPaymentMethod == 'Bank Account') {
const trimmedValue = input.value.trim();
if (input.classList.contains('stripe-donation-routing-number')) { // routing number
if (trimmedValue.length === 9) // if the routing number is 9 digits
return true; // mark as valid
return false;
} else { // account number
if (trimmedValue.length >= 8 && trimmedValue.length <= 17) // if the routing number is 9 digits
return true; // mark as valid
return false;
}
} else { // ACH isn't even selected so don't even need to validate it
return true;
}
}
switch (type) {
case 'email':
const emailRegex = /^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$/;
if (!emailRegex.test(value))
return console.error("validate:", name+':', "Email address is invalid.\n", input, value), false;
return true;
case 'tel':
if (!value || value.length < 10)
return console.error("validate:", name+':', "Phone number is invalid.\n", input, value), false;
return true;
case 'select':
case 'radio':
case 'text':
if (!value || value.length === 0)
return console.error("validate:", name+':', "Field is invalid.\n", input, value), false;
return true;
default:
debug.log("default");
return true;
}
/*if (!value || value.length === 0)
return false;*/
});
return valid;
} catch (error) {
console.error(error);
return false;
}
}, ...lockedProperty
},
DonationInterrupter: {
value: { init: initDonationInterrupter.bind(api) },
enumerable: true,
configurable: true,
writable: true,
}
});
initHooks(api, ['onFrequencyChange', 'onAmountChange', 'onTrySubmit', 'onSubmit']);
api.Frequency.addEventListener('change', (event) => {
if (event.target.checked) {
api.hooks['onFrequencyChange'].forEach((callback) => {
callback.call(api, event.target.value);
});
}
});
api.GiftArray.addEventListeners('change', (event) => {
if (event.target.checked) {
api.hooks['onAmountChange'].forEach((callback) => {
callback.call(api, event.target.value);
});
}
});
api.SubmitButton.addEventListener('click', (event) => {
api.hooks['onTrySubmit'].forEach((callback) => callback.call(api, event));
});
api.root.addEventListener('submit', (event) => {
api.hooks['onSubmit'].forEach((callback) => callback.call(api, event));
});
if (options.makeTabbed)
api.makeTabbed();
/*if (options.fakeSubmit)
window.NA.DonationForm.SubmitButtonCopy = window.NA.DonationForm.SubmitButtonCopy || createNewSubmitButton(window.NA.DonationForm.SubmitButton, { cloneOriginal: false, hideOriginal: true, observeOriginal: false });*/
return api;
}
function createNewSubmitButton (originalSubmitButton = window.NA.DonationForm.SubmitButton, options = {}) {
const defaultOptions = {
cloneOriginal: true,
hideOriginal: true,
observeOriginal: true,
};
options = { ...defaultOptions, ...options };
const newSubmitButton = document.createElement('button');
newSubmitButton.id = "submit-button-copy";
newSubmitButton.classList.add(...originalSubmitButton.classList.values());
newSubmitButton.textContent = originalSubmitButton.value;
originalSubmitButton.after(newSubmitButton);
options.hideOriginal && originalSubmitButton.style.setProperty("display", "none");
return newSubmitButton;
}
function initHooks (api, hookNames = []) {
const hooks = Object.fromEntries(hookNames.map((hookName) => ([hookName, new Array()])));
Object.defineProperty(api, 'hooks', {
value: hooks,
...lockedProperty
});
}
function initDonationInterrupter (options = {}) {
const getExpId = (matching = /Donation Interrupter/) => {
let experiments = window._vwo_exp;
experiments = Object.entries(window._vwo_exp);
let id = experiments.find(([id, data]) => {
const name = data.name,
status = data.status,
variation = data.hasOwnProperty('combination_chosen') ? data.combination_chosen : null;
return variation && status === 'RUNNING' && name.match(matching);
})[1]?.id;
return id;
};
const getExpVariation = (id) => {
let experiment = window._vwo_exp[id];
return experiment.combination_chosen || experiment.combination_selected;
};
const defaultOptions = {
id: [ 'VWO', getExpId(), getExpVariation(getExpId()) ].join('-'),
tokenName: ("NA__THF_DonationInterrupter:" + [ 'VWO', getExpId(), getExpVariation(getExpId()) ].join('-')),
min: 10,
max: 100,
askAmount: (originalAmount) => {
if (originalAmount > 500) // $500+
return false; // don't show
if (originalAmount >= 400) // $400-$500
return 50;
if (originalAmount >= 300) // $300-$399
return 40;
if (originalAmount >= 200) // $200-$299
return 30;
if (originalAmount >= 100) // $100-$199
return 15;
if (originalAmount < 100) // $100-
return 10;
return false;
},
askFrequency: (originalFrequency) => {
return true;
},
popupHTML: {
headingHTML: (`
Lorem ipsum dolor sit amet
`),
bodyHTML: (`
Consectetur adipiscing elit. Sed nec egestas turpis, hendrerit semper nisl. Pellentesque auctor ipsum at
pharetra eleifend. Pellentesque a rhoncus turpis, ut tempus nibh. Donec vel dui hendrerit nisi imperdiet
tincidunt. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos.
Duis efficitur dolor ut nisl blandit imperdiet. Nullam pretium est nunc, tincidunt viverra ligula dapibus.
Duis malesuada:
dui eu venenatis volutpat
urna libero posuere lectus
non tincidunt mauris ligula consectetur felis
lacinia hendrerit enim at molestie
Sed placerat fringilla consequat. Nullam eu pellentesque sem?
This type of book is bound with cardboard that’s covered in cloth, plastic, or leather.
Format
MSRP: $27.00
$21.99
Struggling to break free from the same old conflicts in your relationships? Backed by neuroscience and biblical wisdom, The Mindful Marriage offers a proven, grace-filled approach to healing patterns of pain and strengthening your connection. Whether you’re dating, engaged, married, or in a blended family, it’s ideal for both small-group discussions and couples study.
110 in stock (can be backordered)
Free standard shipping on all orders $75 and up.
Overview
Strengthen Your Connection at Any Stage
Backed by neuroscience and biblical principles, The Mindful Marriage delivers a therapeutic model for couples who want to stop the painful patterns in troubled relationships—or strengthen already healthy ones. Whether you’re dating, engaged, married, or in a blended family, this resource offers practical tools for building deeper connections.
Transformative Restoration Therapy: When Ron and Nan Deal faced the devastating loss of their child, their marriage was nearly shattered by destructive cycles. Seeking help, they turned to pioneering therapists Dr. Terry Hargrave and Sharon Hargrave, LMFT, whose groundbreaking Restoration Therapy has helped millions through the practice of emotional mindfulness. This proven method equips couples to understand themselves in moments of emotional distress and self-regulate for healthier relationships.
Rooted in Science and Biblical Principles: Ron and Nan Deal offer a thoughtful balance of scientifically backed techniques and Scripturally sound teaching, providing practical and spiritual tools to cultivate more loving and faithful marriages.
Great for Small Groups: No extra guide is needed—just read, discuss, and dive into the included exercises together. Share insights, encourage one another, and put the principles into practice. Free bonus materials, including an optional workbook for personal reflection, extra exercises, and articles, are available at RonDeal.org/TMM.
Buy a copy for each couple in your group and start building stronger relationships today!
About the Author
Nan and Ron Deal are coauthors (with Terry & Sharon Hargrave) of the book The Mindful Marriage: Create Your Best Relationship Through Understanding and Managing Yourself. Nan retired from teaching after 25 years in public and private schools. Ron is the author of more than a dozen books and resources, a licensed marriage and family therapist, and a podcaster. Together, Ron and Nan work with various ministries and present marriage enrichment seminars around the country. They have three sons, a daughter-in-law, a grandson, and live in Little Rock, Arkansas.
Ron L. Deal is Director of FamilyLife Blended™ and President of SmartStepfamilies.com. He is a family ministry consultant and conducts marriage and family seminars around the country; he specializes in marriage and stepfamily enrichment. Ron is author of the bestselling book and DVD curriculum The Smart Stepfamily and author/coauthor of The Smart Stepmom, The Smart Stepdad, and The Remarriage Checkup.
He has extensive experience training pastors, ministry leaders, and counselors and has appeared on dozens of national radio and TV broadcasts in the US, Canada, and the UK. Ron is considered a leading voice in stepfamily education and ministry and in the prevention of redivorce. He and his wife, Nan, have been married since 1986 and have three boys. For more information visit RonDeal.org.
Product Details
SKU:
BKH21896
ISBN:
978-1546007388
Publisher:
Worthy Publisher
Language:
English
Page Count:
240
Publication Date:
01/07/2025
Size:
9 × 6 × 0.88 in
Author:
Nan Deal, Ron Deal
Format:
Hardcover
Shipping & Returns
We offer free shipping for orders with subtotals of $75 or greater. You’ll find shipping options and delivery dates at checkout. Please allow 7-10 business days for standard shipping, expedited shipping is available for an additional fee. Orders received after 12 p.m. CST on Fridays will not be processed until the following Monday. You can view our return policy here.
Reviews
There are no reviews yet.
Be the first to review “The Mindful Marriage: Create Your Best Relationship Through Understanding and Managing Yourself” Cancel reply
There are no reviews yet.