Add guest management and GDPR privacy compliance (v0.4.0)
All checks were successful
Create Release Package / build-release (push) Successful in 1m26s
All checks were successful
Create Release Package / build-release (push) Successful in 1m26s
- Create Guest CPT with personal info, address, ID/passport tracking - Add guest-booking integration with AJAX search and linking - Implement GDPR compliance via WordPress Privacy API (export/erasure) - Update EmailNotifier to use Guest CPT data with new placeholders - Add CSS styles for guest search, linked display, and privacy UI Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -577,6 +577,184 @@
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Initialize guest search functionality for booking form.
|
||||
*/
|
||||
function initGuestSearch() {
|
||||
var $searchInput = $('#bnb_booking_guest_search');
|
||||
var $searchResults = $('#bnb-guest-search-results');
|
||||
var $guestIdInput = $('#bnb_booking_guest_id');
|
||||
var $linkedGuestInfo = $('#bnb-linked-guest-info');
|
||||
var $searchContainer = $('#bnb-guest-search-container');
|
||||
var $fieldsContainer = $('#bnb-guest-fields-container');
|
||||
var $unlinkBtn = $('#bnb-unlink-guest');
|
||||
var $guestNameInput = $('#bnb_booking_guest_name');
|
||||
var $guestEmailInput = $('#bnb_booking_guest_email');
|
||||
var $guestPhoneInput = $('#bnb_booking_guest_phone');
|
||||
|
||||
// Exit if not on booking form.
|
||||
if (!$searchInput.length) {
|
||||
return;
|
||||
}
|
||||
|
||||
var searchTimer = null;
|
||||
|
||||
/**
|
||||
* Perform guest search via AJAX.
|
||||
*/
|
||||
function searchGuests() {
|
||||
var query = $searchInput.val().trim();
|
||||
|
||||
if (query.length < 2) {
|
||||
$searchResults.hide().empty();
|
||||
return;
|
||||
}
|
||||
|
||||
$searchResults.html('<div class="bnb-search-loading">' + wpBnbAdmin.i18n.searchingGuests + '</div>').show();
|
||||
|
||||
$.ajax({
|
||||
url: wpBnbAdmin.ajaxUrl,
|
||||
type: 'POST',
|
||||
data: {
|
||||
action: 'wp_bnb_search_guest',
|
||||
nonce: wpBnbAdmin.nonce,
|
||||
search: query
|
||||
},
|
||||
success: function(response) {
|
||||
if (response.success && response.data.guests.length > 0) {
|
||||
var html = '<div class="bnb-guest-search-list">';
|
||||
|
||||
$.each(response.data.guests, function(i, guest) {
|
||||
var isBlocked = guest.status === 'blocked';
|
||||
var statusClass = isBlocked ? 'bnb-guest-blocked' : '';
|
||||
var statusLabel = isBlocked ? ' <span class="bnb-blocked-label">' + wpBnbAdmin.i18n.guestBlocked + '</span>' : '';
|
||||
|
||||
html += '<div class="bnb-guest-search-item ' + statusClass + '" data-guest=\'' + JSON.stringify(guest) + '\'>';
|
||||
html += '<div class="bnb-guest-item-info">';
|
||||
html += '<strong>' + escapeHtml(guest.name) + '</strong>' + statusLabel + '<br>';
|
||||
html += '<small>' + escapeHtml(guest.email || '') + '</small>';
|
||||
if (guest.phone) {
|
||||
html += ' <small>(' + escapeHtml(guest.phone) + ')</small>';
|
||||
}
|
||||
html += '</div>';
|
||||
if (!isBlocked) {
|
||||
html += '<button type="button" class="button button-small bnb-select-guest">' + wpBnbAdmin.i18n.selectGuest + '</button>';
|
||||
}
|
||||
html += '</div>';
|
||||
});
|
||||
|
||||
html += '</div>';
|
||||
$searchResults.html(html);
|
||||
} else {
|
||||
$searchResults.html('<div class="bnb-no-guests">' + wpBnbAdmin.i18n.noGuestsFound + '</div>');
|
||||
}
|
||||
},
|
||||
error: function() {
|
||||
$searchResults.html('<div class="bnb-search-error">' + wpBnbAdmin.i18n.error + '</div>');
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Escape HTML entities.
|
||||
*
|
||||
* @param {string} text Text to escape.
|
||||
* @return {string} Escaped text.
|
||||
*/
|
||||
function escapeHtml(text) {
|
||||
if (!text) return '';
|
||||
var div = document.createElement('div');
|
||||
div.textContent = text;
|
||||
return div.innerHTML;
|
||||
}
|
||||
|
||||
/**
|
||||
* Select a guest from search results.
|
||||
*
|
||||
* @param {Object} guest Guest data.
|
||||
*/
|
||||
function selectGuest(guest) {
|
||||
// Set hidden guest ID.
|
||||
$guestIdInput.val(guest.id);
|
||||
|
||||
// Populate guest fields (for display/fallback).
|
||||
$guestNameInput.val(guest.name).prop('readonly', true);
|
||||
$guestEmailInput.val(guest.email).prop('readonly', true);
|
||||
$guestPhoneInput.val(guest.phone).prop('readonly', true);
|
||||
|
||||
// Update linked guest display.
|
||||
var infoHtml = '<p>';
|
||||
infoHtml += '<span class="dashicons dashicons-admin-users"></span> ';
|
||||
infoHtml += '<strong>' + escapeHtml(guest.name) + '</strong> ';
|
||||
infoHtml += '<a href="' + wpBnbAdmin.ajaxUrl.replace('admin-ajax.php', 'post.php?post=' + guest.id + '&action=edit') + '" target="_blank" class="button button-small">View Guest Profile</a> ';
|
||||
infoHtml += '<button type="button" id="bnb-unlink-guest" class="button button-small button-link-delete">Unlink</button>';
|
||||
infoHtml += '</p>';
|
||||
if (guest.email) {
|
||||
infoHtml += '<p><small>' + escapeHtml(guest.email) + '</small></p>';
|
||||
}
|
||||
|
||||
$linkedGuestInfo.html(infoHtml).show();
|
||||
$searchContainer.hide();
|
||||
$fieldsContainer.hide();
|
||||
$searchResults.hide().empty();
|
||||
$searchInput.val('');
|
||||
|
||||
// Re-bind unlink button.
|
||||
bindUnlinkButton();
|
||||
}
|
||||
|
||||
/**
|
||||
* Unlink guest from booking.
|
||||
*/
|
||||
function unlinkGuest() {
|
||||
$guestIdInput.val('');
|
||||
$guestNameInput.val('').prop('readonly', false);
|
||||
$guestEmailInput.val('').prop('readonly', false);
|
||||
$guestPhoneInput.val('').prop('readonly', false);
|
||||
|
||||
$linkedGuestInfo.hide();
|
||||
$searchContainer.show();
|
||||
$fieldsContainer.show();
|
||||
}
|
||||
|
||||
/**
|
||||
* Bind unlink button event.
|
||||
*/
|
||||
function bindUnlinkButton() {
|
||||
$('#bnb-unlink-guest').off('click').on('click', function(e) {
|
||||
e.preventDefault();
|
||||
unlinkGuest();
|
||||
});
|
||||
}
|
||||
|
||||
// Search input with debounce.
|
||||
$searchInput.on('input', function() {
|
||||
if (searchTimer) {
|
||||
clearTimeout(searchTimer);
|
||||
}
|
||||
searchTimer = setTimeout(searchGuests, 300);
|
||||
});
|
||||
|
||||
// Select guest from results.
|
||||
$searchResults.on('click', '.bnb-select-guest', function(e) {
|
||||
e.preventDefault();
|
||||
var guest = $(this).closest('.bnb-guest-search-item').data('guest');
|
||||
if (guest) {
|
||||
selectGuest(guest);
|
||||
}
|
||||
});
|
||||
|
||||
// Initial unlink button binding.
|
||||
bindUnlinkButton();
|
||||
|
||||
// Close search results when clicking outside.
|
||||
$(document).on('click', function(e) {
|
||||
if (!$(e.target).closest('#bnb_booking_guest_search, #bnb-guest-search-results').length) {
|
||||
$searchResults.hide();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Initialize on document ready.
|
||||
$(document).ready(function() {
|
||||
initLicenseManagement();
|
||||
@@ -586,6 +764,7 @@
|
||||
initPricingMetaBox();
|
||||
initBookingForm();
|
||||
initCalendarPage();
|
||||
initGuestSearch();
|
||||
});
|
||||
|
||||
})(jQuery);
|
||||
|
||||
Reference in New Issue
Block a user