Conditional Logic: Entry Meta

Use the entry meta in conditional logic (e.g. payment status, approval status, etc).

Code

Filename: gw-conditional-logic-entry-meta.php

<?php
/**
 * Gravity Wiz // Gravity Forms // Conditional Logic: Entry Meta
 * https://gravitywiz.com/
 *
 * Supports all registered meta and as well as the "Payment Status" standard meta.
 * Handles enabling conditional logic evaluation on send for GP Notification Scheduler when notification
 * contains a conditional rule for "payment_status".
 *
 * Requires Gravity Forms 2.6.2.
 *
 * Plugin Name:  GF Conditional Logic: Entry Meta
 * Plugin URI:   https://gravitywiz.com/
 * Description:  Use the entry meta in conditional logic (e.g. payment status, approval status, etc).
 * Author:       Gravity Wiz
 * Version:      0.3.1
 * Author URI:   https://gravitywiz.com
 */
class GW_CL_Entry_Meta {

	private static $instance;

	public static function get_instance() {

		if ( ! self::$instance ) {
			self::$instance = new GW_CL_Entry_Meta();
		}

		return self::$instance;
	}

	private function __construct() {

		if ( did_action( 'gform_loaded' ) ) {
			$this->init();
		} else {
			add_action( 'gform_loaded', array( $this, 'init' ) );
		}

	}

	public function init() {

		add_action( 'admin_footer', array( $this, 'output_admin_script' ) );
		add_filter( 'gform_rule_source_value', array( $this, 'set_rule_source_value' ), 10, 5 );
		add_filter( 'gpns_evaluate_conditional_logic_on_send', array( $this, 'gpns_should_evaluate_conditional_logic_on_send' ), 10, 4 );

	}

	public function output_admin_script() {
		if ( wp_script_is( 'gform_form_admin' ) && GFForms::get_page() !== 'form_editor' ) :
			?>
			<script>
				const entryOptions = <?php echo json_encode( $this->get_conditional_logic_options() ); ?>;
				gform.addFilter( 'gform_conditional_logic_fields', function( options, form, selectedFieldId ) {
					for ( const property in entryOptions ) {
						// Entry meta are already added in Notifications and Confirmations conditional logic but not in feeds.
						// Let's just make sure that none of our entry meta options have been previously added.
						if ( entryOptions.hasOwnProperty( property ) && ! options.find( opt => opt.value === entryOptions[ property ].value ) ) {
							options.push( {
								label: entryOptions[ property ].label,
								value: entryOptions[ property ].value
							} );
						}
					}
					return options;
				} );
				gform.addFilter( 'gform_conditional_logic_operators', function( operators, objectType, fieldId ) {
					if ( entryOptions.hasOwnProperty( fieldId ) ) {
						operators = entryOptions[ fieldId ].operators;
					}
					return operators;
				} );
				gform.addFilter( 'gform_conditional_logic_values_input', function( str, objectType, ruleIndex, selectedFieldId, selectedValue ) {
					if ( entryOptions.hasOwnProperty( selectedFieldId ) && entryOptions[ selectedFieldId ].choices ) {
						let inputName = objectType + '_rule_value_' + ruleIndex;
						str = GetRuleValuesDropDown( entryOptions[ selectedFieldId ].choices, objectType, ruleIndex, selectedValue, inputName );
					}
					return str;
				} );
			</script>
			<?php
		endif;
	}

	public function get_conditional_logic_options() {

		$form_ids = 0;

		// Scope entry meta to the current form for GP Email Users and other admin screens
		if ( is_admin() ) {
			if ( rgget( 'page' ) === 'gp-email-users' ) {
				$form_ids = rgpost( '_gform_setting_form' );
				if ( ! $form_ids ) {
					$draft    = gp_email_users()->get_draft();
					$form_ids = $draft['form'];
				}
			}

			if ( GFForms::get_page() && rgget( 'id' ) ) {
				$form_ids = rgget( 'id' );
			}
		}

		$options = array();

		if ( ! empty( $form_ids ) ) {
			$post_submission_conditional_logic_field_types = array(
				'uid' => array(
					'operators' => array(
						'is'          => 'is',
						'isnot'       => 'isNot',
						'>'           => 'greaterThan',
						'<'           => 'lessThan',
						'contains'    => 'contains',
						'starts_with' => 'startsWith',
						'ends_with'   => 'endsWith',
					),
				),
			);
			$form = GFAPI::get_form( is_array( $form_ids ) ? $form_ids[0] : $form_ids );
			if ( $form ) {
				$fields = GFAPI::get_fields_by_type( $form, array_keys( $post_submission_conditional_logic_field_types ) );
				foreach ( $fields as $field ) {
					$options[ $field->id ] = array(
						'label'     => $field->label,
						'value'     => $field->id,
						'operators' => rgars( $post_submission_conditional_logic_field_types, $field->type . '/operators', array() ),
					);
				}
			}
		}

		$entry_meta = GFFormsModel::get_entry_meta( $form_ids );

		$choices_by_key = array(
			'is_approved' => array(
				1 => esc_html__( 'Approved', 'gravityview' ),
				2 => esc_html__( 'Disapproved', 'gravityview' ),
				3 => esc_html__( 'Unapproved', 'gravityview' ),
			),
		);

		foreach ( $entry_meta as $key => $meta ) {
			$options[ $key ] = array(
				'label'     => $meta['label'],
				'value'     => $key,
				'operators' => array(
					'is'    => 'is',
					'isnot' => 'isNot',
				),
			);

			$_choices = rgar( $choices_by_key, $key );

			/*
			 * Use choices by key if exists, otherwise, attempt to pull from $meta/filter/choices, which contains
			 * array{ text: string, value: string }[]
			 */
			if ( empty( $_choices ) ) {
				$filter_choices = rgars( $meta, 'filter/choices' );

				if ( ! empty( $filter_choices ) ) {
					$_choices = wp_list_pluck( $filter_choices, 'text', 'value' );
				}
			}

			if ( ! empty( $_choices ) ) {
				$choices = array();
				foreach ( $_choices as $value => $text ) {
					$choices[] = compact( 'text', 'value' );
				}
				$options[ $key ]['choices'] = $choices;
			}
		}

		$options['payment_status'] = array(
			'label'     => esc_html__( 'Payment Status', 'gravityforms' ),
			'value'     => 'payment_status',
			'choices'   => $this->get_payment_status_choices(),
			'operators' => array(
				'is'    => 'is',
				'isnot' => 'isNot',
			),
		);

		return $options;
	}

	public function get_payment_status_choices() {
		$choices = array();
		foreach ( GFCommon::get_entry_payment_statuses() as $text => $value ) {
			$choices[] = compact( 'text', 'value' );
		}
		return $choices;
	}

	public function set_rule_source_value( $source_value, $rule, $form, $logic, $entry ) {

		$keys   = array_keys( $this->get_conditional_logic_options() );
		$target = $rule['fieldId'];

		if ( in_array( $target, $keys ) && $entry ) {
			if ( in_array( $target, $this->get_runtime_entry_meta_keys(), true ) ) {
				// Some payment add-ons do not update the runtime entry but do update the entry in the database.
				// Fetch the latest from the database.
				$entry = GFAPI::get_entry( $entry['id'] );
			}
			$source_value = rgar( $entry, $rule['fieldId'] );
		}

		return $source_value;
	}

	public function gpns_should_evaluate_conditional_logic_on_send( $eval_on_send, $form, $entry, $notifications ) {

		foreach ( $notifications as $notification ) {
			$_notification = gp_notification_schedule()->get_notification( $form, $notification['nid'] );
			foreach ( rgars( $_notification, 'notification_conditional_logic_object/rules' ) as $rule ) {
				if ( in_array( $rule['fieldId'], $this->get_runtime_entry_meta_keys(), true ) ) {
					return true;
				}
			}
		}

		return $eval_on_send;
	}

	/**
	 * Get the keys for any entry meta that may be updated during a form submission.
	 *
	 * Since these are updated mid-submission, conditional logic in some contexts (like notifications triggered by a feed
	 * action) will be using a stale entry. Identifying these entry meta keys allows us to ensure special functionality
	 * to support them.
	 *
	 * @return mixed|void
	 */
	public function get_runtime_entry_meta_keys() {
		return apply_filters( 'gwclem_runtime_entry_meta_keys', array( 'payment_status' ) );
	}

}

function gw_cl_entry_meta() {
	return GW_CL_Entry_Meta::get_instance();
}

gw_cl_entry_meta();

Leave a Reply

Your email address will not be published. Required fields are marked *

  • Trouble installing this snippet? See our troubleshooting tips.
  • Need to include code? Create a gist and link to it in your comment.
  • Reporting a bug? Provide a URL where this issue can be recreated.

By commenting, I understand that I may receive emails related to Gravity Wiz and can unsubscribe at any time.