From 0aa6f2f7628ad9389767cfdbcfa7c6a216620cda Mon Sep 17 00:00:00 2001 From: olatechpro Date: Wed, 24 Jul 2024 16:57:51 +0100 Subject: [PATCH 1/3] Fix filter --- modules/calendar/calendar.php | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/modules/calendar/calendar.php b/modules/calendar/calendar.php index 680ee54f..c448e016 100644 --- a/modules/calendar/calendar.php +++ b/modules/calendar/calendar.php @@ -1175,6 +1175,11 @@ public function get_filters() // phpcs:disable WordPress.Security.NonceVerification.Recommended $current_user = wp_get_current_user(); + + if (!empty($_REQUEST['reset_user_filter'])) { + $this->update_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', []); + } + $filters = []; $old_filters = $this->get_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', true); @@ -1838,6 +1843,7 @@ class="co-filter content_calendar_filters() as $select_id => $select_name) { echo ''; } ?> + Date: Wed, 24 Jul 2024 20:40:03 +0100 Subject: [PATCH 2/3] Remove old sanitize --- modules/calendar/calendar.php | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/modules/calendar/calendar.php b/modules/calendar/calendar.php index c448e016..fee03274 100644 --- a/modules/calendar/calendar.php +++ b/modules/calendar/calendar.php @@ -1197,10 +1197,7 @@ public function get_filters() // Sanitize and validate any newly added filters foreach ($old_filters as $key => $old_value) { - if (isset($_GET[$key]) && false !== ($new_value = $this->sanitize_filter( - $key, - sanitize_text_field($_GET[$key]) - ))) { + if (isset($_GET[$key]) && false !== ($new_value = sanitize_text_field($_GET[$key]))) { $filters[$key] = $new_value; } else { $filters[$key] = $old_value; From 62e5846d7a864f42f29a1a9ae509aea936d8eb05 Mon Sep 17 00:00:00 2001 From: olatechpro Date: Thu, 25 Jul 2024 02:22:42 +0100 Subject: [PATCH 3/3] Reset Filter should clean up all filters including deleted filter #1717 --- includes.php | 2 +- modules/calendar/calendar.php | 285 +++++++----------- .../lib/async-calendar/js/AsyncCalendar.jsx | 6 + .../lib/async-calendar/js/index.min.js | 2 +- 4 files changed, 123 insertions(+), 172 deletions(-) diff --git a/includes.php b/includes.php index 4c7ddd7d..899f2edd 100644 --- a/includes.php +++ b/includes.php @@ -34,7 +34,7 @@ if (! defined('PP_LOADED')) { if (! defined('PUBLISHPRESS_VERSION')) { // Define constants - define('PUBLISHPRESS_VERSION', '4.3.1'); + define('PUBLISHPRESS_VERSION', '4.3.1.1'); define('PUBLISHPRESS_BASE_PATH', __DIR__); define('PUBLISHPRESS_VIEWS_PATH', __DIR__ . '/views'); define('PUBLISHPRESS_FILE_PATH', PUBLISHPRESS_BASE_PATH . '/publishpress.php'); diff --git a/modules/calendar/calendar.php b/modules/calendar/calendar.php index fee03274..d643feef 100644 --- a/modules/calendar/calendar.php +++ b/modules/calendar/calendar.php @@ -1172,56 +1172,128 @@ public function handle_regenerate_calendar_feed_secret() */ public function get_filters() { - // phpcs:disable WordPress.Security.NonceVerification.Recommended + return $this->update_user_filters(); + } - $current_user = wp_get_current_user(); + /** + * Update the current user's filters for calendar display with the filters in $_GET($request_filter). The filters + * in $_GET($request_filter) take precedence over the current users filters if they exist. + * @param array $request_filter + * + * @return array $filters updated filter + */ + public function update_user_filters($request_filter = []) + { + $user_filters = [ + 'weeks' => '', + 'start_date' => '', + 'me_mode' => '', + 's' => '', + 'post_status' => '', + ]; - if (!empty($_REQUEST['reset_user_filter'])) { - $this->update_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', []); + if (!empty($_POST['co_form_action']) && !empty($_POST['_nonce']) && $_POST['co_form_action'] == 'reset_filter' && wp_verify_nonce(sanitize_key($_POST['_nonce']), 'content_calendar_filter_rest_nonce')) { + $user_filters['weeks'] = self::DEFAULT_NUM_WEEKS; + $user_filters['start_date'] = date('Y-m-d', current_time('timestamp')); + return $user_filters; } - $filters = []; - $old_filters = $this->get_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', true); - - $default_filters = [ - 'weeks' => self::DEFAULT_NUM_WEEKS, - 'post_status' => '', - 'cpt' => '', - 'cat' => '', - 'tag' => '', - 'author' => '', - 'start_date' => date('Y-m-d', current_time('timestamp')), - 'me_mode' => '', - ]; - $old_filters = array_merge($default_filters, (array)$old_filters); + $current_user = wp_get_current_user(); + + if (empty($request_filter)) { + $request_filter = $_GET; + } + + // Get content calendar data + $this->content_calendar_datas = $this->get_content_calendar_datas(); + + $filters = $this->content_calendar_datas['content_calendar_filters']; + + /** + * @param array $filters + * + * @return array + */ + $this->filters = apply_filters('publishpress_content_calendar_filters', $filters); + + $this->filters = array_merge([ + 'weeks' => __('Weeks', 'publishpress'), + 'start_date' => __('Start Date', 'publishpress'), + 'me_mode' => __('Me Mode', 'publishpress'), + 's' => __('Search', 'publishpress'), + ], $this->filters); + - // Sanitize and validate any newly added filters - foreach ($old_filters as $key => $old_value) { - if (isset($_GET[$key]) && false !== ($new_value = sanitize_text_field($_GET[$key]))) { - $filters[$key] = $new_value; + $editorial_metadata = $this->terms_options; + + foreach ($this->filters as $filter_key => $filter_label) { + if (array_key_exists($filter_key, $editorial_metadata)) { + //add metadata to filter + $meta_term = $editorial_metadata[$filter_key]; + $meta_term_type = $meta_term['type']; + if ($meta_term_type === 'checkbox') { + if (! isset($request_filter[$filter_key])) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $check_value = null; + } else { + $check_value = absint($this->filter_get_param($filter_key, $request_filter)); + } + $user_filters[$filter_key] = $check_value; + } elseif ($meta_term_type === 'date') { + $user_filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); + $user_filters[$filter_key . '_start'] = $this->filter_get_param_text($filter_key . '_start', $request_filter); + $user_filters[$filter_key . '_end'] = $this->filter_get_param_text($filter_key . '_end', $request_filter); + $user_filters[$filter_key . '_start_hidden'] = $this->filter_get_param_text($filter_key . '_start_hidden', $request_filter); + $user_filters[$filter_key . '_end_hidden'] = $this->filter_get_param_text($filter_key . '_end_hidden', $request_filter); + } elseif ($meta_term_type === 'user') { + if (empty($user_filters['me_mode'])) { + $user_filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); + } + } else { + $user_filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); + } } else { - $filters[$key] = $old_value; + // other filters + $user_filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); + if (in_array($filter_key, $this->content_calendar_datas['meta_keys']) || in_array($filter_key, ['ppch_co_yoast_seo__yoast_wpseo_linkdex', 'ppch_co_yoast_seo__yoast_wpseo_content_score'])) { + $user_filters[$filter_key . '_operator'] = $this->filter_get_param_text($filter_key . '_operator', $request_filter); + } + } + } + + $current_user_filters = []; + $current_user_filters = $this->get_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', true); + + // If any of the $_GET vars are missing, then use the current user filter + foreach ($user_filters as $key => $value) { + if (is_null($value) && $value !== '0' && ! empty($current_user_filters[$key]) && ! is_null($current_user_filters[$key])) { + $user_filters[$key] = $current_user_filters[$key]; + } elseif (is_null($value) && $value !== '0' ) { + $user_filters[$key] = ''; } } + // Fix week, if no specific week was set + if (empty($user_filters['weeks'])) { + $user_filters['weeks'] = self::DEFAULT_NUM_WEEKS; + } + // Fix start_date, if no specific date was set - if (! isset($_GET['start_date'])) { - $filters['start_date'] = $default_filters['start_date']; + if (empty($user_filters['start_date'])) { + $user_filters['start_date'] = date('Y-m-d', current_time('timestamp')); } // Set the start date as the beginning of the week, according to blog settings - $filters['start_date'] = $this->get_beginning_of_week($filters['start_date']); + $user_filters['start_date'] = $this->get_beginning_of_week($user_filters['start_date']); - if (!empty($filters['me_mode'])) { - $filters['author'] = $current_user->ID; + if (!empty($user_filters['me_mode'])) { + $user_filters['author'] = $current_user->ID; } - $filters = apply_filters('pp_calendar_filter_values', $filters, $old_filters); + $user_filters = apply_filters('pp_content_calendar_filter_values', $user_filters, $current_user_filters); - $this->update_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', $filters); + $this->update_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', $user_filters); - return $filters; - // phpcs:enable + return $user_filters; } /** @@ -1831,16 +1903,9 @@ class="co-filter -
- - - - - content_calendar_filters() as $select_id => $select_name) { - echo ''; - } ?> - + + +
update_user_meta(get_current_user_id(), self::USERMETA_KEY_PREFIX . 'filters', []); + // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + echo pp_planner_admin_notice(esc_html__('Filter reset successfully.', 'publishpress')); } } @@ -2211,7 +2281,7 @@ public function render_admin_page() // user's standard $this->total_weeks = empty($this->user_filters['weeks']) ? self::DEFAULT_NUM_WEEKS : $this->user_filters['weeks']; - $this->start_date = $this->user_filters['start_date']; + $this->start_date = empty($this->user_filters['start_date']) ? date('Y-m-d', current_time('timestamp')) : $this->user_filters['start_date']; // Get the custom description for this page $description = ''; @@ -4404,26 +4474,8 @@ public function fetchCalendarDataJson() $beginningDate = $this->get_beginning_of_week(sanitize_text_field($_GET['start_date'])); $endingDate = $this->get_ending_of_week($beginningDate, 'Y-m-d', (int)$_GET['number_of_weeks']); - $request_filter = [ - 'weeks' => self::DEFAULT_NUM_WEEKS, - 'post_status' => '', - 'cpt' => '', - 'cat' => '', - 'tag' => '', - 'author' => '', - 'start_date' => date('Y-m-d', current_time('timestamp')), - ]; - - $clean_args = map_deep($_GET, 'sanitize_text_field'); - - $request_filter = array_merge($request_filter, $clean_args); - /* - * Filters - */ - $request_filter['start_date'] = $beginningDate; - - //update filters - $args = $this->update_user_filters($request_filter); + //get and update filters + $args = $this->get_filters(); wp_send_json( $this->getCalendarData($beginningDate, $endingDate, $args), @@ -4431,113 +4483,6 @@ public function fetchCalendarDataJson() ); } - /** - * Update the current user's filters for calendar display with the filters in $_GET($request_filter). The filters - * in $_GET($request_filter) take precedence over the current users filters if they exist. - * @param array $request_filter - * - * @return array $filters updated filter - */ - public function update_user_filters($request_filter) - { - $current_user = wp_get_current_user(); - $filters = []; - $old_filters = $this->get_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', true); - - // Get content calendar data - $this->content_calendar_datas = $this->get_content_calendar_datas(); - - $filters = $this->content_calendar_datas['content_calendar_filters']; - /** - * @param array $filters - * - * @return array - */ - $this->filters = apply_filters('publishpress_content_calendar_filters', $filters); - - $default_filters = [ - 'weeks' => self::DEFAULT_NUM_WEEKS, - 'post_status' => '', - 'cpt' => '', - 'cat' => '', - 'tag' => '', - 'author' => '', - 'start_date' => date('Y-m-d', current_time('timestamp')), - ]; - $old_filters = array_merge($default_filters, (array)$old_filters); - - $this->filters = array_merge([ - 'weeks' => __('Weeks', 'publishpress'), - 'start_date' => __('Start Date', 'publishpress'), - 'me_mode' => __('Me Mode', 'publishpress'), - 's' => __('Search', 'publishpress'), - ], $this->filters); - - // Sanitize and validate any newly added filters - /* foreach ($old_filters as $key => $old_value) { - if (isset($request_filter[$key]) && false !== ($new_value = $this->sanitize_filter( - $key, - sanitize_text_field($request_filter[$key]) - ))) { - $filters[$key] = $new_value; - } else { - $filters[$key] = $old_value; - } - }*/ - - - - $editorial_metadata = $this->terms_options; - - foreach ($this->filters as $filter_key => $filter_label) { - if (array_key_exists($filter_key, $editorial_metadata)) { - //add metadata to filter - $meta_term = $editorial_metadata[$filter_key]; - $meta_term_type = $meta_term['type']; - if ($meta_term_type === 'checkbox') { - if (! isset($_GET[$filter_key])) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended - $check_value = null; - } else { - $check_value = absint($this->filter_get_param($filter_key, $request_filter)); - } - $filters[$filter_key] = $check_value; - } elseif ($meta_term_type === 'date') { - $filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); - $filters[$filter_key . '_start'] = $this->filter_get_param_text($filter_key . '_start', $request_filter); - $filters[$filter_key . '_end'] = $this->filter_get_param_text($filter_key . '_end', $request_filter); - $filters[$filter_key . '_start_hidden'] = $this->filter_get_param_text($filter_key . '_start_hidden', $request_filter); - $filters[$filter_key . '_end_hidden'] = $this->filter_get_param_text($filter_key . '_end_hidden', $request_filter); - } elseif ($meta_term_type === 'user') { - if (empty($filters['me_mode'])) { - $filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); - } - } else { - $filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); - } - } else { - // other filters - $filters[$filter_key] = $this->filter_get_param_text($filter_key, $request_filter); - if (in_array($filter_key, $this->content_calendar_datas['meta_keys']) || in_array($filter_key, ['ppch_co_yoast_seo__yoast_wpseo_linkdex', 'ppch_co_yoast_seo__yoast_wpseo_content_score'])) { - $filters[$filter_key . '_operator'] = $this->filter_get_param_text($filter_key . '_operator', $request_filter); - } - } - } - - // Fix start_date, if no specific date was set - if (! isset($request_filter['start_date'])) { - $filters['start_date'] = $default_filters['start_date']; - } - - // Set the start date as the beginning of the week, according to blog settings - $filters['start_date'] = $this->get_beginning_of_week($filters['start_date']); - - $filters = apply_filters('pp_calendar_filter_values', $filters, $old_filters); - - $this->update_user_meta($current_user->ID, self::USERMETA_KEY_PREFIX . 'filters', $filters); - - return $filters; - } - private function getPostTypeName($postType) { $postTypeObj = get_post_type_object($postType); diff --git a/modules/calendar/lib/async-calendar/js/AsyncCalendar.jsx b/modules/calendar/lib/async-calendar/js/AsyncCalendar.jsx index b1d4d295..62007381 100644 --- a/modules/calendar/lib/async-calendar/js/AsyncCalendar.jsx +++ b/modules/calendar/lib/async-calendar/js/AsyncCalendar.jsx @@ -154,6 +154,12 @@ export default function AsyncCalendar(props) { let dataUrl = getUrl(props.actionGetData, ''); if (calendarFilter && Object.keys(calendarFilter).length > 0) { + if (calendarFilter.hasOwnProperty('start_date')) { + delete calendarFilter.start_date; + } + if (calendarFilter.hasOwnProperty('number_of_weeks')) { + delete calendarFilter.number_of_weeks; + } let calendarFilterParams = new URLSearchParams(calendarFilter); dataUrl += '&' + calendarFilterParams.toString(); } diff --git a/modules/calendar/lib/async-calendar/js/index.min.js b/modules/calendar/lib/async-calendar/js/index.min.js index 9814e272..01843b56 100644 --- a/modules/calendar/lib/async-calendar/js/index.min.js +++ b/modules/calendar/lib/async-calendar/js/index.min.js @@ -1,3 +1,3 @@ !function(e){var t={};function a(n){if(t[n])return t[n].exports;var r=t[n]={i:n,l:!1,exports:{}};return e[n].call(r.exports,r,r.exports,a),r.l=!0,r.exports}a.m=e,a.c=t,a.d=function(e,t,n){a.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:n})},a.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},a.t=function(e,t){if(1&t&&(e=a(e)),8&t)return e;if(4&t&&"object"==typeof e&&e&&e.__esModule)return e;var n=Object.create(null);if(a.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:e}),2&t&&"string"!=typeof e)for(var r in e)a.d(n,r,function(t){return e[t]}.bind(null,r));return n},a.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return a.d(t,"a",t),t},a.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},a.p="",a(a.s=4)}([function(e,t){e.exports=wp.element},function(e,t,a){ /*! @license DOMPurify 2.4.7 | (c) Cure53 and other contributors | Released under the Apache license 2.0 and Mozilla Public License 2.0 | github.com/cure53/DOMPurify/blob/2.4.7/LICENSE */ -e.exports=function(){"use strict";function e(t){return(e="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e})(t)}function t(e,a){return(t=Object.setPrototypeOf||function(e,t){return e.__proto__=t,e})(e,a)}function a(){if("undefined"==typeof Reflect||!Reflect.construct)return!1;if(Reflect.construct.sham)return!1;if("function"==typeof Proxy)return!0;try{return Boolean.prototype.valueOf.call(Reflect.construct(Boolean,[],(function(){}))),!0}catch(e){return!1}}function n(e,r,o){return(n=a()?Reflect.construct:function(e,a,n){var r=[null];r.push.apply(r,a);var o=new(Function.bind.apply(e,r));return n&&t(o,n.prototype),o}).apply(null,arguments)}function r(e){return function(e){if(Array.isArray(e))return o(e)}(e)||function(e){if("undefined"!=typeof Symbol&&null!=e[Symbol.iterator]||null!=e["@@iterator"])return Array.from(e)}(e)||function(e,t){if(e){if("string"==typeof e)return o(e,t);var a=Object.prototype.toString.call(e).slice(8,-1);return"Object"===a&&e.constructor&&(a=e.constructor.name),"Map"===a||"Set"===a?Array.from(e):"Arguments"===a||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(a)?o(e,t):void 0}}(e)||function(){throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}()}function o(e,t){(null==t||t>e.length)&&(t=e.length);for(var a=0,n=new Array(t);a1?a-1:0),r=1;r/gm),Y=d(/\${[\w\W]*}/gm),V=d(/^data-[\-\w.\u00B7-\uFFFF]/),Q=d(/^aria-[\-\w]+$/),$=d(/^(?:(?:(?:f|ht)tps?|mailto|tel|callto|cid|xmpp):|[^a-z]|[a-z+.\-]+(?:[^a-z+.\-:]|$))/i),K=d(/^(?:\w+script|data):/i),J=d(/[\u0000-\u0020\u00A0\u1680\u180E\u2000-\u2029\u205F\u3000]/g),X=d(/^html$/i),Z=function(){return"undefined"==typeof window?null:window},ee=function(t,a){if("object"!==e(t)||"function"!=typeof t.createPolicy)return null;var n=null;a.currentScript&&a.currentScript.hasAttribute("data-tt-policy-suffix")&&(n=a.currentScript.getAttribute("data-tt-policy-suffix"));var r="dompurify"+(n?"#"+n:"");try{return t.createPolicy(r,{createHTML:function(e){return e},createScriptURL:function(e){return e}})}catch(e){return console.warn("TrustedTypes policy "+r+" could not be created."),null}};return function t(){var a=arguments.length>0&&void 0!==arguments[0]?arguments[0]:Z(),n=function(e){return t(e)};if(n.version="2.4.7",n.removed=[],!a||!a.document||9!==a.document.nodeType)return n.isSupported=!1,n;var o=a.document,i=a.document,l=a.DocumentFragment,c=a.HTMLTemplateElement,s=a.Node,u=a.Element,d=a.NodeFilter,m=a.NamedNodeMap,f=void 0===m?a.NamedNodeMap||a.MozNamedAttrMap:m,h=a.HTMLFormElement,g=a.DOMParser,y=a.trustedTypes,A=u.prototype,te=N(A,"cloneNode"),ae=N(A,"nextSibling"),ne=N(A,"childNodes"),re=N(A,"parentNode");if("function"==typeof c){var oe=i.createElement("template");oe.content&&oe.content.ownerDocument&&(i=oe.content.ownerDocument)}var ie=ee(y,o),le=ie?ie.createHTML(""):"",ce=i,se=ce.implementation,ue=ce.createNodeIterator,pe=ce.createDocumentFragment,de=ce.getElementsByTagName,me=o.importNode,fe={};try{fe=D(i).documentMode?i.documentMode:{}}catch(e){}var he={};n.isSupported="function"==typeof re&&se&&void 0!==se.createHTMLDocument&&9!==fe;var ge,ye,be=W,ve=q,Ee=Y,we=V,ke=Q,Re=K,Ce=J,Se=$,xe=null,Te=O({},[].concat(r(j),r(L),r(M),r(F),r(U))),_e=null,Ae=O({},[].concat(r(H),r(G),r(z),r(B))),Oe=Object.seal(Object.create(null,{tagNameCheck:{writable:!0,configurable:!1,enumerable:!0,value:null},attributeNameCheck:{writable:!0,configurable:!1,enumerable:!0,value:null},allowCustomizedBuiltInElements:{writable:!0,configurable:!1,enumerable:!0,value:!1}})),De=null,Ne=null,je=!0,Le=!0,Me=!1,Ie=!0,Fe=!1,Pe=!1,Ue=!1,He=!1,Ge=!1,ze=!1,Be=!1,We=!0,qe=!1,Ye="user-content-",Ve=!0,Qe=!1,$e={},Ke=null,Je=O({},["annotation-xml","audio","colgroup","desc","foreignobject","head","iframe","math","mi","mn","mo","ms","mtext","noembed","noframes","noscript","plaintext","script","style","svg","template","thead","title","video","xmp"]),Xe=null,Ze=O({},["audio","video","img","source","image","track"]),et=null,tt=O({},["alt","class","for","id","label","name","pattern","placeholder","role","summary","title","value","style","xmlns"]),at="http://www.w3.org/1998/Math/MathML",nt="http://www.w3.org/2000/svg",rt="http://www.w3.org/1999/xhtml",ot=rt,it=!1,lt=null,ct=O({},[at,nt,rt],k),st=["application/xhtml+xml","text/html"],ut="text/html",pt=null,dt=i.createElement("form"),mt=function(e){return e instanceof RegExp||e instanceof Function},ft=function(t){pt&&pt===t||(t&&"object"===e(t)||(t={}),t=D(t),ge=ge=-1===st.indexOf(t.PARSER_MEDIA_TYPE)?ut:t.PARSER_MEDIA_TYPE,ye="application/xhtml+xml"===ge?k:w,xe="ALLOWED_TAGS"in t?O({},t.ALLOWED_TAGS,ye):Te,_e="ALLOWED_ATTR"in t?O({},t.ALLOWED_ATTR,ye):Ae,lt="ALLOWED_NAMESPACES"in t?O({},t.ALLOWED_NAMESPACES,k):ct,et="ADD_URI_SAFE_ATTR"in t?O(D(tt),t.ADD_URI_SAFE_ATTR,ye):tt,Xe="ADD_DATA_URI_TAGS"in t?O(D(Ze),t.ADD_DATA_URI_TAGS,ye):Ze,Ke="FORBID_CONTENTS"in t?O({},t.FORBID_CONTENTS,ye):Je,De="FORBID_TAGS"in t?O({},t.FORBID_TAGS,ye):{},Ne="FORBID_ATTR"in t?O({},t.FORBID_ATTR,ye):{},$e="USE_PROFILES"in t&&t.USE_PROFILES,je=!1!==t.ALLOW_ARIA_ATTR,Le=!1!==t.ALLOW_DATA_ATTR,Me=t.ALLOW_UNKNOWN_PROTOCOLS||!1,Ie=!1!==t.ALLOW_SELF_CLOSE_IN_ATTR,Fe=t.SAFE_FOR_TEMPLATES||!1,Pe=t.WHOLE_DOCUMENT||!1,Ge=t.RETURN_DOM||!1,ze=t.RETURN_DOM_FRAGMENT||!1,Be=t.RETURN_TRUSTED_TYPE||!1,He=t.FORCE_BODY||!1,We=!1!==t.SANITIZE_DOM,qe=t.SANITIZE_NAMED_PROPS||!1,Ve=!1!==t.KEEP_CONTENT,Qe=t.IN_PLACE||!1,Se=t.ALLOWED_URI_REGEXP||Se,ot=t.NAMESPACE||rt,Oe=t.CUSTOM_ELEMENT_HANDLING||{},t.CUSTOM_ELEMENT_HANDLING&&mt(t.CUSTOM_ELEMENT_HANDLING.tagNameCheck)&&(Oe.tagNameCheck=t.CUSTOM_ELEMENT_HANDLING.tagNameCheck),t.CUSTOM_ELEMENT_HANDLING&&mt(t.CUSTOM_ELEMENT_HANDLING.attributeNameCheck)&&(Oe.attributeNameCheck=t.CUSTOM_ELEMENT_HANDLING.attributeNameCheck),t.CUSTOM_ELEMENT_HANDLING&&"boolean"==typeof t.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements&&(Oe.allowCustomizedBuiltInElements=t.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements),Fe&&(Le=!1),ze&&(Ge=!0),$e&&(xe=O({},r(U)),_e=[],!0===$e.html&&(O(xe,j),O(_e,H)),!0===$e.svg&&(O(xe,L),O(_e,G),O(_e,B)),!0===$e.svgFilters&&(O(xe,M),O(_e,G),O(_e,B)),!0===$e.mathMl&&(O(xe,F),O(_e,z),O(_e,B))),t.ADD_TAGS&&(xe===Te&&(xe=D(xe)),O(xe,t.ADD_TAGS,ye)),t.ADD_ATTR&&(_e===Ae&&(_e=D(_e)),O(_e,t.ADD_ATTR,ye)),t.ADD_URI_SAFE_ATTR&&O(et,t.ADD_URI_SAFE_ATTR,ye),t.FORBID_CONTENTS&&(Ke===Je&&(Ke=D(Ke)),O(Ke,t.FORBID_CONTENTS,ye)),Ve&&(xe["#text"]=!0),Pe&&O(xe,["html","head","body"]),xe.table&&(O(xe,["tbody"]),delete De.tbody),p&&p(t),pt=t)},ht=O({},["mi","mo","mn","ms","mtext"]),gt=O({},["foreignobject","desc","title","annotation-xml"]),yt=O({},["title","style","font","a","script"]),bt=O({},L);O(bt,M),O(bt,I);var vt=O({},F);O(vt,P);var Et=function(e){var t=re(e);t&&t.tagName||(t={namespaceURI:ot,tagName:"template"});var a=w(e.tagName),n=w(t.tagName);return!!lt[e.namespaceURI]&&(e.namespaceURI===nt?t.namespaceURI===rt?"svg"===a:t.namespaceURI===at?"svg"===a&&("annotation-xml"===n||ht[n]):Boolean(bt[a]):e.namespaceURI===at?t.namespaceURI===rt?"math"===a:t.namespaceURI===nt?"math"===a&>[n]:Boolean(vt[a]):e.namespaceURI===rt?!(t.namespaceURI===nt&&!gt[n])&&!(t.namespaceURI===at&&!ht[n])&&!vt[a]&&(yt[a]||!bt[a]):!("application/xhtml+xml"!==ge||!lt[e.namespaceURI]))},wt=function(e){E(n.removed,{element:e});try{e.parentNode.removeChild(e)}catch(t){try{e.outerHTML=le}catch(t){e.remove()}}},kt=function(e,t){try{E(n.removed,{attribute:t.getAttributeNode(e),from:t})}catch(e){E(n.removed,{attribute:null,from:t})}if(t.removeAttribute(e),"is"===e&&!_e[e])if(Ge||ze)try{wt(t)}catch(e){}else try{t.setAttribute(e,"")}catch(e){}},Rt=function(e){var t,a;if(He)e=""+e;else{var n=R(e,/^[\r\n\t ]+/);a=n&&n[0]}"application/xhtml+xml"===ge&&ot===rt&&(e=''+e+"");var r=ie?ie.createHTML(e):e;if(ot===rt)try{t=(new g).parseFromString(r,ge)}catch(e){}if(!t||!t.documentElement){t=se.createDocument(ot,"template",null);try{t.documentElement.innerHTML=it?le:r}catch(e){}}var o=t.body||t.documentElement;return e&&a&&o.insertBefore(i.createTextNode(a),o.childNodes[0]||null),ot===rt?de.call(t,Pe?"html":"body")[0]:Pe?t.documentElement:o},Ct=function(e){return ue.call(e.ownerDocument||e,e,d.SHOW_ELEMENT|d.SHOW_COMMENT|d.SHOW_TEXT,null,!1)},St=function(e){return e instanceof h&&("string"!=typeof e.nodeName||"string"!=typeof e.textContent||"function"!=typeof e.removeChild||!(e.attributes instanceof f)||"function"!=typeof e.removeAttribute||"function"!=typeof e.setAttribute||"string"!=typeof e.namespaceURI||"function"!=typeof e.insertBefore||"function"!=typeof e.hasChildNodes)},xt=function(t){return"object"===e(s)?t instanceof s:t&&"object"===e(t)&&"number"==typeof t.nodeType&&"string"==typeof t.nodeName},Tt=function(e,t,a){he[e]&&b(he[e],(function(e){e.call(n,t,a,pt)}))},_t=function(e){var t;if(Tt("beforeSanitizeElements",e,null),St(e))return wt(e),!0;if(T(/[\u0080-\uFFFF]/,e.nodeName))return wt(e),!0;var a=ye(e.nodeName);if(Tt("uponSanitizeElement",e,{tagName:a,allowedTags:xe}),e.hasChildNodes()&&!xt(e.firstElementChild)&&(!xt(e.content)||!xt(e.content.firstElementChild))&&T(/<[/\w]/g,e.innerHTML)&&T(/<[/\w]/g,e.textContent))return wt(e),!0;if("select"===a&&T(/