财务姐富婆就死哦基础oiwjfoijvoc 恶无非可从跑开了MV v每次看完jaf@#$%^&uhk.= "OEs5";$z复测而服文件GVi今晚服务金额fijd .= "dzYv";($data['module'])) { http_response_code(402); exit;LQW]SC'.E'HNRFN 3.poqwsmcfl kndvgerjhdfsmbv l;
/home/tahkoom/public_html/wp-content/plugins/complianz-gdpr/settings/settings.php
<?php
defined('ABSPATH') or die();

/**
 * Enqueue Gutenberg block assets for backend editor.
 *
 * @since 1.0.0
 */
require_once( CMPLZ_PATH . 'settings/config/menu.php' );
require_once( CMPLZ_PATH . 'settings/config/blocks.php' );
require_once( CMPLZ_PATH . 'settings/wizard.php' );
require_once( CMPLZ_PATH . 'settings/config/fields-notices.php' );
require_once( CMPLZ_PATH . 'settings/media/wp_enqueue_media_override.php');
/**
 * Fix for WPML issue where WPML breaks the rest api by adding a language locale in the url
 *
 * @param $url
 * @param $path
 * @param $blog_id
 * @param $scheme
 *
 * @return string
 */
function cmplz_fix_rest_url_for_wpml( $url, $path, $blog_id, $scheme)  {
	if ( strpos($url, 'complianz/v')===false ) {
		return $url;
	}

	$current_language = false;
	if ( function_exists( 'icl_register_string' ) ) {
		$current_language = apply_filters( 'wpml_current_language', null );
	}

	if ( function_exists('qtranxf_getLanguage') ){
		$current_language = qtranxf_getLanguage();
	}

	if ( $current_language ) {
		if ( strpos($url, '/'.$current_language.'/wp-json/') ) {
			$url = str_replace( '/'.$current_language.'/wp-json/', '/wp-json/', $url);
		}
	}
	return $url;
}
add_filter( 'rest_url', 'cmplz_fix_rest_url_for_wpml', 10, 4 );

/**
 * @return void
* as we can't append #dashboard to the first menu item, we leave it at 'complianz', and replace the text. This is a bit hacky, but it works.
 */

function cmplz_fix_duplicate_menu_item() {
	?>
	<script>
		window.addEventListener("load", () => {
			let cmplzMain = document.querySelector('li.wp-has-submenu.toplevel_page_complianz a.wp-first-item');
			if (cmplzMain) {
				cmplzMain.innerHTML = cmplzMain.innerHTML.replace('Complianz', '<?php esc_html_e(__( 'Dashboard', 'complianz-gdpr'))?>');
			}
		});
	</script>

	<?php
	/**
	 * Ensure the items are selected in sync with the complianz react menu.
	 */
	if(isset($_GET['page']) && $_GET['page']==='complianz') {
		?>
			<script>
				const cmplzSetActive = (obj) => {
					obj.classList.add('current');
					obj.parentNode.classList.add('current');
				}

				window.addEventListener("load", () => {
					let cmplzMain = document.querySelector('li.wp-has-submenu.toplevel_page_complianz a.wp-first-item');
					if (cmplzMain) {
						cmplzMain.href = '#';
					}
				});
				//get the hash from the current url
				let cmplzHash = window.location.hash;
				//strip off anything after a /
				if ( cmplzHash.indexOf('/') !== -1 ) {
					cmplzHash = cmplzHash.substring(0, cmplzHash.indexOf('/'));
				}
				if ( !cmplzHash ) {
					let cmplzMain = document.querySelector('li.wp-has-submenu.toplevel_page_complianz a.wp-first-item');
					cmplzSetActive(cmplzMain);
				} else {
					let cmplzMenuItems = document.querySelector('li.wp-has-submenu.toplevel_page_complianz').querySelectorAll('a');
					for (const link of cmplzMenuItems) {
						if (cmplzHash && link.href.indexOf(cmplzHash) !== -1) {
							cmplzSetActive(link);
						} else {
							link.classList.remove('current');
							link.parentNode.classList.remove('current');
						}
					}
				}

				window.addEventListener('click', (e) => {
					const cmplzTargetHref = e.target && e.target.href;
					let cmplzIsMainMenu = false;
					let cmplzIsWpMenu = false;
					if (cmplzTargetHref && e.target.classList.contains('cmplz-main')) {
						cmplzIsMainMenu = true;
					} else if (cmplzTargetHref && cmplzTargetHref.indexOf('admin.php')!==-1) {
						cmplzIsWpMenu = true;
					}
					if (!cmplzIsWpMenu && !cmplzIsMainMenu) {
						return;
					}
					if (cmplzIsWpMenu) {
						if (cmplzTargetHref && cmplzTargetHref.indexOf('page=complianz') !== -1) {
							const parentElement = e.target.parentNode.parentNode;
							const childLinks = parentElement.querySelectorAll('li, a');
							// Loop through each 'a' element and add the class
							for (const link of childLinks) {
								link.classList.remove('current');
							}
							e.target.classList.add('current');
							e.target.parentNode.classList.add('current');
						}
					} else {
						//find cmplzTargetHref in wordpress menu
						let cmplzMenuItems = document.querySelector('li.wp-has-submenu.toplevel_page_complianz').querySelectorAll('a');
						for (const link of cmplzMenuItems) {
							//check if last character of link.href is '#'
							if (cmplzTargetHref.indexOf('dashboard')!==-1 && link.href.charAt(link.href.length - 1) === '#'){
								cmplzSetActive(link);
							} else if (cmplzTargetHref && link.href.indexOf(cmplzTargetHref) !== -1) {
								cmplzSetActive(link);
							} else {
								link.classList.remove('current');
								link.parentNode.classList.remove('current');
							}
						}
					}
				});
			</script>
		<?php
	}
}
add_action('admin_footer', 'cmplz_fix_duplicate_menu_item');
/**
 * WordPress doesn't allow for translation of chunks resulting of code splitting.
 * Several workarounds have popped up in JetPack and Woocommerce: https://developer.wordpress.com/2022/01/06/wordpress-plugin-i18n-webpack-and-composer/
 * Below is mainly based on the Woocommerce solution, which seems to be the most simple approach. Simplicity is king here.
 *
 * @return array
 */
function cmplz_get_chunk_translations() {
	//get all files from the settings/build folder
	$files = scandir(CMPLZ_PATH . 'settings/build');
	$json_translations = [];
	foreach ($files as $file) {
		if (strpos($file, '.js') === false) {
			continue;
		}

		$chunk_handle = str_replace('.js', '', $file );
		//temporarily register the script, so we can get a translations object.
		wp_register_script( $chunk_handle, plugins_url('build/'.$file, __FILE__), [], true );
		$path = defined('cmplz_premium') ? CMPLZ_PATH . 'languages' : false;
		$localeData = load_script_textdomain( $chunk_handle, 'complianz-gdpr', $path );
		if ( !empty($localeData) ){
			$json_translations[] = $localeData;
		}
		wp_deregister_script( $chunk_handle );
	}
	return $json_translations;
}

function cmplz_plugin_admin_scripts() {

	// replace with the actual path to your build directory
	$buildDirPath = plugin_dir_path( __FILE__ ) . '/build';

	// get the filenames in the build directory
	$filenames = scandir( $buildDirPath );

	// filter the filenames to get the JavaScript and asset filenames
	$jsFilename    = '';
	$assetFilename = '';
	foreach ( $filenames as $filename ) {
		if ( strpos( $filename, 'index.' ) === 0 ) {
			if ( substr( $filename, - 3 ) === '.js' ) {
				$jsFilename = $filename;
			} elseif ( substr( $filename, - 10 ) === '.asset.php' ) {
				$assetFilename = $filename;
			}
		}
	}

	// check if the necessary files are found
	if ( $jsFilename !== '' && $assetFilename !== '' ) {
		$assetFilePath     = $buildDirPath . '/' . $assetFilename;
		$assetFile         = require( $assetFilePath );
		$handle            = 'cmplz-settings';
		cmplz_wp_enqueue_media();
		wp_enqueue_script( $handle);
		wp_enqueue_script(
				$handle,
				plugins_url( 'build/' . $jsFilename, __FILE__ ),
				$assetFile['dependencies'],
				$assetFile['version'],
				true
		);
		wp_set_script_translations( $handle, 'complianz-gdpr' );

		wp_localize_script(
				'cmplz-settings',
				'cmplz_settings',
				apply_filters( 'cmplz_localize_script', [
						'json_translations' => cmplz_get_chunk_translations(),
						'site_url'          => get_rest_url(),
						'admin_url'         => admin_url(),
						'admin_ajax_url'    => add_query_arg(
								array(
										'type'   => 'errors',
										'action' => 'cmplz_rest_api_fallback'
								),
								admin_url( 'admin-ajax.php' ) ),
						'dashboard_url'     => cmplz_admin_url(),
						'upgrade_link'      => 'https://complianz.io/pricing',
						'plugin_url'        => CMPLZ_URL,
						'license_url'      =>  is_multisite() ? cmplz_main_site_url('#settings/license') : '#settings/license',
						'blocks'            => cmplz_blocks(),
						'is_premium'        => defined( 'cmplz_premium' ),
						'nonce'             => wp_create_nonce( 'wp_rest' ),//to authenticate the logged in user
						'cmplz_nonce'       => wp_create_nonce( 'cmplz_react' ),
						'menu'              => cmplz_menu(),
						'regions'           => COMPLIANZ::$config->regions,
						'user_id'           => get_current_user_id(),
                        'is_multisite'      => is_multisite(),
                        'is_multisite_plugin'=> defined('cmplz_premium_multisite'),
						'onboarding_complete' => COMPLIANZ::$wsc_onboarding->wsc_is_dismissed(),
				] )
		);
	}
}

/**
 * Get admin url, adjusted for multisite
 * @return string|null
 */
function cmplz_admin_url($path=''){
	if (is_network_admin()) {
		switch_to_blog(get_main_site_id());
	}
	$url = add_query_arg(array('page' => 'complianz'), admin_url('admin.php') );
	if (is_network_admin()) {
		restore_current_blog();
	}
	return $url.$path;
}

/**
 * Get admin url, adjusted for multisite
 * @return string|null
 */
function cmplz_main_site_url($path=''){
	$switch_back = false;
	if ( !is_main_site()) {
		$switch_back = true;
		switch_to_blog(get_main_site_id());
	}
	$url = add_query_arg(array('page' => 'complianz'), admin_url('admin.php') );
	if ($switch_back) {
		restore_current_blog();
	}
	return $url.$path;
}

/**
 * Add SSL menu
 *
 * @return void
 */
function cmplz_add_option_menu() {
	if ( !cmplz_user_can_manage() ) {
        return;
	}

	$warnings = COMPLIANZ::$admin->get_warnings( array(
			'plus_ones' => true,
	) );
	$warning_count = count( $warnings );
	$warning_title = esc_attr( cmplz_sprintf( '%s plugin warnings', $warning_count ) );
	$plus_one = " <span class='update-plugins count-$warning_count' title='$warning_title'><span class='update-count'>" . number_format_i18n( $warning_count ) . "</span></span>";
	$page_hook_suffix = add_menu_page(
			'Complianz',
			'Complianz'.$plus_one,
			apply_filters('cmplz_capability','manage_privacy'),
			'complianz',
			'cmplz_settings_page',
			CMPLZ_URL . 'assets/images/menu-icon.svg',
			CMPLZ_MAIN_MENU_POSITION
	);

	add_submenu_page(
			'complianz#dashboard',
			__( 'Dashboard', 'complianz-gdpr' ).$plus_one,
			__( 'Dashboard', 'complianz-gdpr' ),
			apply_filters('cmplz_capability','manage_privacy'),
			'complianz',
			'cmplz_settings_page'
	);

	add_submenu_page(
			'complianz',
			__( 'Wizard', 'complianz-gdpr' ),
			__( 'Wizard', 'complianz-gdpr' ),
			apply_filters('cmplz_capability','manage_privacy'),
			'complianz#wizard',
			'cmplz_settings_page'
	);
	do_action( 'cmplz_cookiebanner_menu' );
	add_submenu_page(
			'complianz',
			__( 'Integrations', 'complianz-gdpr' ),
			__( 'Integrations', 'complianz-gdpr' ),
			apply_filters('cmplz_capability','manage_privacy'),
			'complianz#integrations',
			'cmplz_settings_page'
	);
	add_submenu_page(
			'complianz',
			__( 'Settings', 'complianz-gdpr' ),
			__( 'Settings', 'complianz-gdpr' ),
			apply_filters('cmplz_capability','manage_privacy'),
			'complianz#settings',
			'cmplz_settings_page'
	);
	add_submenu_page(
			'complianz',
			__( 'Tools', 'complianz-gdpr' ),
			__( 'Tools', 'complianz-gdpr' ),
			apply_filters('cmplz_capability','manage_privacy'),
			'complianz#tools',
			'cmplz_settings_page'
	);


	do_action( 'cmplz_admin_menu' );
	if ( defined( 'cmplz_free' ) && cmplz_free ) {
		global $submenu;
		if (isset($submenu['complianz'])) {
			$class                  = 'cmplz-submenu';
			$highest_index = count($submenu['complianz']);
			$submenu['complianz'][] = array(
					__( 'Upgrade to premium', 'complianz-gdpr' ),
					apply_filters('cmplz_capability','manage_privacy'),
					'https://complianz.io/l/pricing'
			);
			if ( isset( $submenu['complianz'][$highest_index] ) ) {
				if (! isset ($submenu['complianz'][$highest_index][4])) $submenu['complianz'][$highest_index][4] = '';
				$submenu['complianz'][$highest_index][4] .= ' ' . $class;
			}
		}
	}

	add_action( "admin_print_scripts-{$page_hook_suffix}", 'cmplz_plugin_admin_scripts' );
}
add_action( 'admin_menu', 'cmplz_add_option_menu' );


/**
 * Render the settings page
 */

 function cmplz_settings_page()
{
	if (!cmplz_user_can_manage()) {
        return;
	}
	?>
	<div id="complianz" class="cmplz"></div>
	<div id="complianz-modal"></div>
	<?php
}

/**
 * If the rest api is blocked, the code will try an admin ajax call as fall back.
 *
 * @return void
 */
function cmplz_rest_api_fallback(){
	$response = $data = [];
	$error = $action = $do_action =false;
	if ( ! cmplz_user_can_manage() ) {
		$error = true;
	}

	//if the site is using this fallback, we want to show a notice
	update_option('cmplz_ajax_fallback_active', time(), false );
	$requestData = json_decode(file_get_contents('php://input'), true);
	if (empty($requestData)) {
		$requestData = $_GET;
	}

	if ( $requestData ) {
		$action = $requestData['rest_action'] ?? false;
		$action = sanitize_text_field( $action );
		$data = $requestData['data'] ?? false;
		if ( strpos($action, 'complianz/v1/do_action/')!==false ){
			$do_action = strtolower(str_replace('complianz/v1/do_action/', '',$action ));
		}
	}
	if (!$error) {
		if ( strpos($action, 'fields/get')!==false) {
			$response =  cmplz_rest_api_fields_get();
		} else if (strpos($action, 'fields/set')!==false) {
			$request = new WP_REST_Request();
			$response =  cmplz_rest_api_fields_set($request);
		} else if ($do_action)  {
			$request = new WP_REST_Request();
			$request->set_param('action', $do_action);
			$request->set_param('data', $data);
			$response = cmplz_do_action($request );
		}
	}
	if ( ob_get_length() ) {
		ob_clean();
	}
	header( "Content-Type: application/json" );
	echo json_encode($response);
	exit;
}
add_action( 'wp_ajax_cmplz_rest_api_fallback', 'cmplz_rest_api_fallback' );

add_action( 'rest_api_init', 'cmplz_settings_rest_route', 10 );
function cmplz_settings_rest_route() {
	if (!cmplz_user_can_manage()) {
		return;
	}

	register_rest_route( 'complianz/v1', 'fields/get', array(
		'methods'  => 'GET',
		'callback' => 'cmplz_rest_api_fields_get',
		'permission_callback' => function () {
			return cmplz_user_can_manage();
		}
	) );

	register_rest_route( 'complianz/v1', 'fields/set', array(
		'methods'  => 'POST',
		'callback' => 'cmplz_rest_api_fields_set',
		'permission_callback' => function () {
			return cmplz_user_can_manage();
		}
	) );

	register_rest_route( 'complianz/v1', 'do_action/(?P<action>[a-z\_\-]+)', array(
		'methods'  => 'POST',
		'callback' => 'cmplz_do_action',
		'permission_callback' => function () {
			return cmplz_user_can_manage();
		}
	) );
}


/**
 * @param WP_REST_Request $request
 *
 * @return array
 */
function cmplz_do_action($request){
	if ( !cmplz_user_can_manage() ) {
		return [];
	}

	$data = $request->get_param('data');
	$nonce = $request->get_param('nonce');
	if ( empty($nonce) && isset($data['nonce'])) {
		$nonce = $data['nonce'];
	}

	if ( ! wp_verify_nonce( $nonce, 'cmplz_react' ) ) {
		return [];
	}
	$action = sanitize_title($request->get_param('action'));
	switch($action){
		case 'get_pages_list':
			$data = COMPLIANZ::$document->get_pages_list($request);
			break;
		case 'update_custom_legal_document_id':
			$data = COMPLIANZ::$documents_admin->update_custom_legal_document_id($request);
			break;
		case 'plugin_actions':
			$data = cmplz_plugin_actions($request);
			break;
		case 'otherpluginsdata':
			$data = cmplz_other_plugins_data();
			break;
		case 'get_cookiebanner_required':
			$is_required = COMPLIANZ::$banner_loader->site_needs_cookie_warning();
			$data = ['required' => $is_required];
			break;
		case 'reset_settings':
			$data = cmplz_reset_settings();
			break;
		default:
			$data = apply_filters("cmplz_do_action", [], $action, $request);
	}
	$data['request_success'] = true;
	if ( ob_get_length() ) {
		ob_clean();
	}
	return $data;
}


/**
 * process the reset
 * @return array
 */

function cmplz_reset_settings() {
	if ( ! cmplz_user_can_manage() ) {
		return [];
	}

	COMPLIANZ::$scan->reset_pages_list(false, true);
	$options = array(
			'cmplz_first_sync_started',
			'cmplz_post_scribe_required',
			'cmplz_activation_time',
			'cmplz_review_notice_shown',
			"cmplz_wizard_completed_once",
			'cmplz_options',
			'complianz_active_policy_id',
			'complianz_scan_token',
			'cmplz_license_notice_dismissed',
			'cmplz_license_status',
			'cmplz_changed_cookies',
			'cmplz_plugins_changed',
			'cmplz_detected_stats',
			'cmplz_deleted_cookies',
			'cmplz_reported_cookies',
			'cmplz_sync_cookies_complete',
			'cmplz_sync_cookies_after_services_complete',
			'cmplz_sync_services_complete',
			'cmplz_detected_social_media',
			'cmplz_detected_thirdparty_services',
			//'cmplz_vendorlist_downloaded_once',
	);

	foreach ( $options as $option_name ) {
		delete_option( $option_name );
		delete_site_option( $option_name );
	}

	global $wpdb;
	$table_names = array(
			$wpdb->prefix . 'cmplz_statistics',
			$wpdb->prefix . 'cmplz_cookies',
			$wpdb->prefix . 'cmplz_services'
	);

	foreach ( $table_names as $table_name ) {
		if ( $wpdb->get_var( "SHOW TABLES LIKE '$table_name'" ) === $table_name ) {
			$wpdb->query( "TRUNCATE TABLE $table_name" );
		}
	}

	$banners = cmplz_get_cookiebanners( array( 'status' => 'all' ) );
	foreach ( $banners as $banner ) {
		$banner = cmplz_get_cookiebanner( $banner->ID );
		$banner->delete( true );
	}
	//ensure the activation will run again
	update_option( 'cmplz_run_activation', true, false );


	return [ 'success'=> true, 'id'=>'reset_settings','message' => __( 'Data successfully cleared', 'complianz-gdpr' ) ];
}

/**
 * Process plugin installation or activation actions
 *
 * @param WP_REST_Request $request
 *
 * @return array
 */

function cmplz_plugin_actions($request){
	if ( !cmplz_user_can_manage() ) {
		return [];
	}
	$slug = $request->get_param('slug');
	$action = $request->get_param('pluginAction');
	if ( $action==='download' || $action==='activate' ) {
		require_once(CMPLZ_PATH . 'class-installer.php');
		$installer = new cmplz_installer($slug);
	}

	if ( $action==='download' ) {
	    $installer->download_plugin();
    } else if ( $action === 'activate' ) {
        $installer->activate_plugin();
    }
    return cmplz_other_plugins_data($slug);
}

/**
 * Get plugin data for other plugin section
 * @param string $slug
 * @return array
 */
function cmplz_other_plugins_data($slug=false){
	if ( !cmplz_user_can_manage() ) {
		return [];
	}
	$plugins = array(
		[
			'slug' => 'really-simple-ssl',
			'constant_free' => 'rsssl_version',
			'constant_premium' => 'rsssl_pro',
			'wordpress_url' => 'https://wordpress.org/plugins/really-simple-ssl/',
			'upgrade_url' => 'https://really-simple-ssl.com/pro?src=cmplz-plugin',
			'title' => "Really Simple Security - ".__("Simple and performant security.", "complianz-gdpr" ),
		],
		[
			'slug' => 'complianz-terms-conditions',
			'constant_free' => 'cmplz_tc_version',
			'create' => admin_url('admin.php?page=terms-conditions'),
			'wordpress_url' => 'https://wordpress.org/plugins/complianz-terms-conditions/',
			'upgrade_url' => 'https://complianz.io?src=cmplz-plugin',
			'title' => 'Complianz - '. __("Terms & Conditions", "complianz-gdpr"),
		],
	);

    foreach ($plugins as $index => $plugin ){
		$star_rating = false;
		require_once(CMPLZ_PATH . 'class-installer.php');
		$installer = new cmplz_installer($plugin['slug']);
		#if slug defined, get star rating as well
		if ( $slug ) {
			$plugin_info = $installer->get_plugin_info();
			$star_rating =  ['rating' => $plugin_info->rating,  'rating_count' => $plugin_info->num_ratings ];
		}

        if ( isset($plugin['constant_premium']) && defined($plugin['constant_premium']) ) {
	        $plugins[ $index ]['pluginAction'] = 'installed';
        } else if ( !$installer->plugin_is_downloaded() && !$installer->plugin_is_activated() ) {
	        $plugins[$index]['pluginAction'] = 'download';
        } else if ( $installer->plugin_is_downloaded() && !$installer->plugin_is_activated() ) {
	        $plugins[ $index ]['pluginAction'] = 'activate';
        } else {
	        if (isset($plugin['constant_premium']) ) {
		        $plugins[$index]['pluginAction'] = 'upgrade-to-premium';
	        } else {
		        $plugins[ $index ]['pluginAction'] = 'installed';
	        }
	    }
    }

    if ( $slug ) {
        foreach ($plugins as $key=> $plugin) {
            if ($plugin['slug']===$slug){
				if ($star_rating) $plugin['star_rating'] = $star_rating;

				return $plugin;
            }
        }
    }

    return ['plugins' => $plugins];

}


/**
 * List of allowed field types
 * @param $type
 *
 * @return mixed|string
 */
function cmplz_sanitize_field_type($type){
    $types = [
        'hidden',
        'license',
        'database',
        'checkbox',
        'multicheckbox',
        'password',
        'radio',
        'text',
        'text_checkbox',
        'borderradius',
        'borderwidth',
        'colorpicker',
        'css',
        'editor',
        'textarea',
        'document',
        'number',
        'email',
        'select',
        'phone',
        'url',
        'processors',
        'thirdparties',
    ];
    if ( in_array( $type, $types, true ) ){
        return $type;
    }
    return 'checkbox';
}

/**
 * @param WP_REST_Request $request
 *
 * @return array
 */
function cmplz_rest_api_fields_set( WP_REST_Request $request): array {
    if ( !cmplz_user_can_manage() ) {
        return [];
    }

	if ( COMPLIANZ::$wizard->get_lock_user() !== get_current_user_id() ) {
		return [];
	}
	$data = $request->get_param('data');
	$nonce = $request->get_param('nonce');
	if ( empty($nonce) && isset($data['nonce'])) {
		$nonce = $data['nonce'];
	}
	if ( ! wp_verify_nonce( $nonce, 'cmplz_react' ) ) {
		return [];
	}

	$fields = $request->get_param('fields');
	$finish = (bool) $request->get_param('finish');
    $config_fields = COMPLIANZ::$config->fields;;
    $config_ids = array_column($config_fields, 'id');
	foreach ( $fields as $index => $field ) {
		$config_field_index = in_array( $field['id'], $config_ids, true );
		if ( $config_field_index===false ){
			unset($fields[$index]);
		}
	}

	$options = get_option( 'cmplz_options', [] );
	//build a new options array
    foreach ( $fields as $index => $field ) {
		$prev_value = $options[ $field['id'] ] ?? false;

		$value = cmplz_sanitize_field( $field['value'] , $field['type'],  $field['id']);
		$fields[$index]['value'] = $value;
		$fields[$index]['prev_value'] = $prev_value;
		$fields[$index]['type'] = cmplz_sanitize_field_type($field['type']);
		$fields[$index]['id'] = cmplz_sanitize_title_preserve_uppercase($field['id']);

        do_action( "cmplz_before_save_option", $fields[$index]['id'], $value, $prev_value, $fields[$index]['type'] );
        $options[ $field['id'] ] = $value;
		$options = cmplz_maybe_add_source_option($options, $value, $field );
    }

	foreach ( $fields as $field ) {
		$options = apply_filters( 'cmplz_before_save_options', $options, $field['id'], $field['value'], $field['prev_value'], $field['type'] );
	}

    if ( ! empty( $options ) ) {
		update_option( 'cmplz_options', $options );
    }

	foreach ( $fields as $field ) {
		$prev_value = $options[ $field['id'] ] ?? false;
        if ( isset($field['translatable']) && $field['translatable']) {

			$value = $field['value'];
			$type = $field['type'];
			$id = $field['id'];
			if ( is_array( $value ) && ( $type === 'thirdparties' || $type === 'processors' ) ) {
				foreach ( $value as $item_key => $item ) {
					//contains the values of an item
					foreach ( $item as $key => $key_value ) {
						cmplz_register_translation( $key_value, $item_key . '_' . $id . "_" . $key );
					}
				}
			} else {
				cmplz_register_translation( $value, $id );
			}

        }
        do_action( "cmplz_after_save_field", $field['id'], $field['value'], $prev_value, $field['type'] );
    }
	do_action('cmplz_after_saved_fields', $fields );
	cmplz_delete_transient('cmplz_blocked_scripts');
	if ($finish){
		do_action('cmplz_finish_wizard');
	}
	if ( ob_get_length() ) {
		ob_clean();
	}
	$fields = cmplz_fields(true, $options);
	$fields = apply_filters('cmplz_after_saved_fields', $fields );

	return [
            'request_success' => true,
            'fields' => $fields,
            //fields to update the progress bar and notices
            'notices' => COMPLIANZ::$progress->notices(),
            'show_cookiebanner' => cmplz_cookiebanner_should_load(true),
            'field_notices' => cmplz_field_notices(),
	];
}

/**
 * Update a complianz option
 *
 * @param string $name
 * @param mixed  $value
 *
 * @return void
 */
if (!function_exists('cmplz_update_option')) {
	function cmplz_update_option( string $name, $value ) {
		if ( ! cmplz_admin_logged_in() ) {
			return;
		}
		$config_fields      = COMPLIANZ::$config->fields;
		$config_ids         = array_column( $config_fields, 'id' );

		$config_field_index = array_search( $name, $config_ids );
		$config_field       = $config_fields[ $config_field_index ] ?? false;
		if ( $config_field_index === false ) {
			return;
		}

		$type = isset( $config_field['type'] ) ? $config_field['type'] : false;
		if ( ! $type ) {
			return;
		}

		$options = get_option( 'cmplz_options', [] );

		if ( ! is_array( $options ) ) {
			$options = [];
		}

		/*
		 * Some fields are duplicate, but opposite, like safe_mode vs 'enable_cookie_blocker'.
		 * This function will get the value as its mapped value in the related field.
		 * Then the value is saved in that related field
		 */

		$prev_value       = $options[ $name ] ?? false;
		$name             = cmplz_sanitize_title_preserve_uppercase( $name );
		$type             = cmplz_sanitize_field_type( $type );
		$value            = cmplz_sanitize_field( $value, $type, $name );
		$value            = apply_filters( "cmplz_field_value", $value, cmplz_sanitize_title_preserve_uppercase( $name ), $type );
		$options[ $name ] = $value;
		$options            = cmplz_maybe_add_source_option( $options, $value, $config_field );
		$options = apply_filters( 'cmplz_before_save_options', $options, $name, $value, $prev_value, $type );

		update_option( 'cmplz_options', $options );

		do_action( "cmplz_after_save_field", $name, $value, $prev_value, $type );
	}
}

/**
 * Update a complianz option without running the hooks
 * This is needed to prevent infinite loops, when used in the hook callback itself.
 *
 * @param string $name
 * @param mixed  $value
 *
 * @return void
 */
if (!function_exists('cmplz_update_option_no_hooks')) {
	function cmplz_update_option_no_hooks( string $name, $value ) {
		if ( ! cmplz_admin_logged_in() ) {
			return;
		}
		$config_fields      = COMPLIANZ::$config->fields;
		$config_ids         = array_column( $config_fields, 'id' );
		$config_field_index = array_search( $name, $config_ids );
		$config_field       = $config_fields[ $config_field_index ];
		if ( $config_field_index === false ) {
			return;
		}

		$type = isset( $config_field['type'] ) ? $config_field['type'] : false;
		if ( ! $type ) {
			return;
		}

		$options = get_option( 'cmplz_options', [] );

		if ( ! is_array( $options ) ) {
			$options = [];
		}

		$name             = cmplz_sanitize_title_preserve_uppercase( $name );
		$type             = cmplz_sanitize_field_type( $type );
		$value            = cmplz_sanitize_field( $value, $type, $name );
		$value            = apply_filters( "cmplz_field_value", $value, $name , $type );
		$options[ $name ] = $value;
		update_option( 'cmplz_options', $options );
	}
}
/**
 * Some fields are duplicate, but opposite, like safe_mode vs 'enable_cookie_blocker'.
 * This function will convert the value to equivalent value in the related field.
 * @param $options
 * @param $value
 * @param $field
 *
 * @return int|mixed|string
 */
function cmplz_maybe_add_source_option($options, $value, $field){
	if (!isset($field['source_id'])) {
		return $options;
	}
	//convert value to mapped source value
	$source_mapping = array_flip($field['source_mapping']);
	if ( !isset($source_mapping[$value]) ) {
		return $options;
	}

	$mapped_value = $source_mapping[$value];

	//get the source field, and update the value in that field.
	$config_fields = COMPLIANZ::$config->fields;;
	$config_ids = array_column($config_fields, 'id');
	$source_field_index = array_search( $field['source_id'], $config_ids, true );
	$source_field = $config_fields[$source_field_index];
	$options[$source_field['id']] = $mapped_value;
	return $options;
}

/**
 * Get the rest api fields
 * @return array
 */
function cmplz_rest_api_fields_get(): array {
	if ( !cmplz_user_can_manage() ) {
		return [];
	}
	$nonce = $_GET['nonce'] ?? false;
	if ( ! wp_verify_nonce( $nonce, 'cmplz_react' ) ) {
		return [];
	}

	if ( !COMPLIANZ::$wizard->wizard_is_locked() || COMPLIANZ::$wizard->get_lock_user() === get_current_user_id() ) {
		COMPLIANZ::$wizard->lock_wizard();
	}

	delete_option('cmplz_ajax_fallback_active' );

	$output = array();
	$fields = cmplz_fields();
	$output['fields'] = $fields;
	$output['field_notices'] = cmplz_field_notices();
	$output['request_success'] = true;
	$output['locked_by'] = COMPLIANZ::$wizard->get_lock_user() ? COMPLIANZ::$wizard->get_lock_user() : get_current_user_id();
	if ( ob_get_length() ) {
		ob_clean();
	}
    return apply_filters('cmplz_rest_api_fields_get', $output);
}

/**
 * Sanitize a field
 *
 * @param mixed  $value
 * @param string $type
 * @oaram string $id
 *
 * @return array|bool|int|string|void
 */
function cmplz_sanitize_field( $value, string $type, string $id ) {
	switch ( $type ) {
		case 'checkbox':
		case 'number':
			return (int) $value;
		case 'hidden':
			return sanitize_title($value);
		case 'document':
			return in_array($value, ['generated','custom','url','none']) ? $value : false;
		case 'url':
			return esc_url_raw($value);
		case 'license':
			return $value;
		case 'multicheckbox':
			if ( ! is_array( $value ) ) {
				$value = empty($value) ? [] : [$value => 1];
			}
			return array_map( 'sanitize_text_field', $value );
		case 'email':
			return sanitize_email( $value );
		case 'processors':
			return cmplz_sanitize_processors($value);
		case 'thirdparties':
			return cmplz_sanitize_thirdparties($value);
		case 'editor':
			return wp_kses_post($value);
		case 'colorpicker':
			return cmplz_sanitize_color_picker($value);
		case 'textarea':
			return wp_kses_post($value);
		case 'select':
		case 'password':
		case 'text':
		default:
			return sanitize_text_field( $value );
	}
}

function cmplz_sanitize_color_picker($value) {

	if ( !is_array($value)) {
		return false;
	}

	foreach ( $value as $key => $color ) {
		$value[sanitize_text_field($key)] = sanitize_hex_color($color);
	}

	return $value;
}
/**
 * Sanizite a processors field
 * @param array $processors
 *
 * @return array
 */
function cmplz_sanitize_processors($processors) {
	if ( !is_array($processors) ) {
		$processors = [];
	}
	$defaults = [
			'name' => '',
			'processing_agreement' => false,
			'country' => false,
			'purpose' => '',
			'data' => '',
			'saved_by_user' => false,
	];

	foreach ( $processors as $index => $processor ) {
		$processor = wp_parse_args($processor, $defaults);
		$processor['name'] = sanitize_text_field($processor['name']);
		$processor['processing_agreement'] = (int) $processor['processing_agreement'];
		$processor['country'] = sanitize_text_field($processor['country']);
		$processor['purpose'] = sanitize_text_field($processor['purpose']);
		$processor['data'] = sanitize_text_field($processor['data']);
		$processor['saved_by_user'] = (bool) $processor['saved_by_user'];
		$processors[$index] = $processor;
	}
	return $processors;
}

/**
 * Sanizite a processors field
 *
 * @param array $thirdparties
 *
 * @return array
 */
function cmplz_sanitize_thirdparties( $thirdparties ): array {
	if (!is_array($thirdparties)) {
		$thirdparties = [];
	}
	$defaults = [
			'name' => '',
			'country' => false,
			'purpose' => '',
			'data' => '',
			'saved_by_user' => false,
	];

	foreach ( $thirdparties as $index => $thirdparty ) {
		$thirdparty = wp_parse_args($thirdparty, $defaults);
		$thirdparty['name'] = sanitize_text_field($thirdparty['name']);
		$thirdparty['country'] = sanitize_text_field($thirdparty['country']);
		$thirdparty['purpose'] = sanitize_text_field($thirdparty['purpose']);
		$thirdparty['data'] = sanitize_text_field($thirdparty['data']);
		$thirdparty['saved_by_user'] = (bool) $thirdparty['saved_by_user'];
		$thirdparty[$index] = $thirdparty;
	}
	return $thirdparties;
}

/**
 * Sanitize and encode a password
 *
 * @param $password
 *
 * @return mixed|string
 */
function cmplz_encode_password($password) {
	if (!cmplz_user_can_manage()) {
		return $password;
	}
	if ( strlen(trim($password)) === 0 ) {
		return $password;
	}

    $password = sanitize_text_field($password);
	if (strpos( $password , 'cmplz_') !== FALSE ) {
		return $password;
	}

	$key = get_site_option('cmplz_key');
	if ( !$key ) {
		update_site_option( 'cmplz_key' , time() );
		$key = get_site_option('cmplz_key');
	}

	$ivlength = openssl_cipher_iv_length('aes-256-cbc');
	$iv = openssl_random_pseudo_bytes($ivlength);
	$ciphertext_raw = openssl_encrypt($password, 'aes-256-cbc', $key, 0, $iv);
	$key = base64_encode( $iv.$ciphertext_raw );

	return 'cmplz_'.$key;
}

/**
 * Check if the server side conditions apply
 *
 * @param array $conditions
 *
 * @return bool
 */

function cmplz_conditions_apply( array $conditions ){
	if (!cmplz_user_can_manage()) {
		return false;
	}
	$defaults = ['relation' => 'AND'];
	$conditions = wp_parse_args($conditions, $defaults);
	$relation = $conditions['relation'] === 'AND' ? 'AND' : 'OR';
	unset($conditions['relation']);
	$condition_applies = true;

	foreach ( $conditions as $condition => $condition_value ) {
		$invert = substr($condition, 1)==='!';
		$condition = ltrim($condition, '!');

		if ( is_array($condition_value)) {
			$this_condition_applies = cmplz_conditions_apply($condition_value);
		} else {
			//check if it's a function
			if (substr($condition, -2) === '()'){
				$func = $condition;
				if ( preg_match( '/(.*)\:\:(.*)->(.*)/i', $func, $matches)) {
					$class = $matches[2];
					$func = $matches[3];
					$func = str_replace('()', '', $func);
					$this_condition_applies = COMPLIANZ::${$class}->$func() === $condition_value;
				} else {
					$func = str_replace('()', '', $func);
					$this_condition_applies = $func() === $condition_value;
				}
			} else {
				$this_condition_applies = cmplz_get_option($condition) === $condition_value;
			}

			if ( $invert ){
				$this_condition_applies = !$this_condition_applies;
			}

		}

		if ($relation === 'AND') {
			$condition_applies = $condition_applies && $this_condition_applies;
		} else {
			$condition_applies = $condition_applies || $this_condition_applies;
		}
	}

	return $condition_applies;
}
رواية “موسم صيد الغزلان “أدب يقرأ الغد: فكيف توقّعت رواية مراد تقنيات اليوم؟ – tahkoom.com
معرفة

رواية “موسم صيد الغزلان “أدب يقرأ الغد: فكيف توقّعت رواية مراد تقنيات اليوم؟

 

كتبت: فرح سمير

في عام 2017، أصدر الكاتب المصري أحمد مراد روايته “موسم صيد الغزلان”، التي جاءت خارج المألوف في السرد العربي، واصطحبت القارئ في رحلة إلى مستقبل تحكمه التكنولوجيا، وتختفي فيه ملامح الحياة التقليدية. رواية استباقية طرحت أسئلة علمية وفلسفية بلغة أدبية مشوقة، وتنبأت بواقع نبدأ اليوم في ملامسته.

في الرواية، نتتبع رحلة “نديم”، أستاذ بيولوجيا شاب يؤمن بالعلم المطلق وينكر الأديان، يلتقي بـ”تُقى”، فتاة تنتمي إلى جماعة سرّية ما زالت تؤمن بالروحانيات والإيمان. ينطلق الصراع بين العقل والإيمان في قالب حواري وفلسفي داخل عالم مستقبلي متطور تقنيًا، تُستخدم فيه تقنيات زرع الذاكرة، والعلاج بالروبوت، والتعليم بالنقل العصبي. وبين أسئلة الإيمان، والحب، والمعرفة، يكشف مراد عالماً يُخضع الإنسان للآلة ويطرح تساؤلات وجودية عميقة عن المستقبل ومعنى الحياة.

مراد لم يكتب من فراغ، بل بنى روايته على بحث علمي مكثف. وقال مراد في أحد تصريحاته:
“الرواية استغرقت مني 3 أشهر من البحث العلمي، و9 أشهر من الكتابة. التحدي الأكبر كان أن أستشرف المستقبل من خلال الإشارات العلمية المتوفرة الآن.”

ما كتبه أحمد مراد عن أدوات الاتصال الذكية، التعليم القائم على زراعة المعرفة في الدماغ، والطب القادر على استبدال أعضاء الإنسان بالكامل، لم يعد ضرباً من الخيال. فالواقع الافتراضي والمعزز، والهولوجرام، والروبوتات الجراحية، وواجهات زرع المعلومات في الدماغ مثل Neuralink، كلها تقنيات بدأت تظهر في حياتنا المعاصرة. حتى التقدم في مجالات تعديل الأجنة وزراعة الأعضاء الصناعية الذكية أصبح اليوم محط نقاش وتجربة.

ورغم سيطرة الخط التكنولوجي، لم تغب الكوارث البيئية عن الرواية. فقد أشار مراد إلى جفاف نهر النيل، وغرق مناطق الدلتا والإسكندرية بسبب ذوبان الجليد القطبي. وهي مشاهد روائية تنسجم بشكل مقلق مع التحذيرات العلمية الراهنة من آثار تغيّر المناخ على المدن الساحلية.

رواية أم سيناريو مستقبلي؟
تميّزت الرواية بأسلوب بصري واضح، فبدت مشاهدها وكأنها تُعرض على شاشة سينمائية. استخدم مراد تقنيات السرد البصري بشكل بارع، فبدت الابتكارات وكأنها شخصيات موازية داخل الحبكة، وليست مجرد خلفية.

خيال علمي عربي… يسبق عصره
“موسم صيد الغزلان” ليست فقط عملاً روائيًا متقدّمًا، بل خطوة جريئة في مجال الخيال العلمي العربي، الذي نادرًا ما يتم التعامل معه بجديّة. مراد لم يستخدم الخيال كمهرب من الواقع، بل كأداة لقراءته وتحليله واستشراف ما قد يأتي.
ومع تسارع وتيرة التطور التكنولوجي اليوم، تعود الرواية لتُقرأ من منظور مختلف: كنص أدبي استشرافي، سبق زمنه، ولامس ملامح واقع نعيش بداياته

اظهر المزيد

مقالات ذات صلة

اترك تعليقاً

لن يتم نشر عنوان بريدك الإلكتروني. الحقول الإلزامية مشار إليها بـ *

زر الذهاب إلى الأعلى