b6df4dbb92
MLS plugin fixes from this session: - Fix silent insert failures: location column NOT NULL was rejecting wpdb->insert calls, causing ~18k new properties since Dec 2025 to be lost. Inserts now build raw SQL with ST_PointFromText so the spatial column is populated atomically. - Auto-refresh expired media URLs in MLS_Media_Handler::fetch_and_cache(), guarded by a property-level GET_LOCK so concurrent fetches share one API refresh. - Normalize WP_Error to null in mls_get_property_image() so callers can rely on the documented string|null contract. - Support comma-separated property_type filters in MLS_Query and MLS_Cluster so the homepage "View All Commercial" link (?property_type=Commercial+Sale,Land,Farm) actually filters correctly. - Incremental sync now looks back 10 minutes past the latest modification timestamp as a safety margin against missed records. - Smart sync exits silently (info-level, not warning) when a full sync is in progress. Operational: - New cron: weekly full sync Sundays at 3 AM (/usr/local/bin/mls-full-sync). - New cron: hourly 2GB cap on mls-thumbnails/ and cache/transformed-images/ (/usr/local/bin/mls-image-cache-cap). - Logrotate config for wp-content/debug.log (2-day retention, daily rotation, delaycompress). Repo policy: - CLAUDE.md updated with explicit "commit everything except build artifacts" policy. - .gitignore: untrack runtime image caches and debug.log rotations. Other modifications in this snapshot are pre-existing in-flight theme/plugin/db_content_updates work. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
193 lines
6.0 KiB
PHP
Executable File
193 lines
6.0 KiB
PHP
Executable File
<?php
|
|
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit;
|
|
}
|
|
|
|
/**
|
|
* WPForms widget.
|
|
*
|
|
* @since 1.0.2
|
|
*/
|
|
class WPForms_Widget extends WP_Widget {
|
|
|
|
/**
|
|
* Hold widget settings defaults, populated in constructor.
|
|
*
|
|
* @since 1.0.2
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $defaults;
|
|
|
|
/**
|
|
* Constructor.
|
|
*
|
|
* @since 1.0.2
|
|
*/
|
|
public function __construct() {
|
|
|
|
// Widget defaults.
|
|
$this->defaults = [
|
|
'title' => '',
|
|
'form_id' => '',
|
|
'show_title' => false,
|
|
'show_desc' => false,
|
|
];
|
|
|
|
// Widget Slug.
|
|
$widget_slug = 'wpforms-widget';
|
|
|
|
// Widget basics.
|
|
$widget_ops = [
|
|
'classname' => $widget_slug,
|
|
'description' => esc_html_x( 'Display a form.', 'Widget', 'wpforms-lite' ),
|
|
'show_instance_in_rest' => false,
|
|
];
|
|
|
|
// Widget controls.
|
|
$control_ops = [
|
|
'id_base' => $widget_slug,
|
|
];
|
|
|
|
// Load widget.
|
|
parent::__construct( $widget_slug, esc_html_x( 'WPForms', 'Widget', 'wpforms-lite' ), $widget_ops, $control_ops );
|
|
}
|
|
|
|
/**
|
|
* Output the HTML for this widget.
|
|
*
|
|
* @since 1.0.2
|
|
*
|
|
* @param array $args An array of standard parameters for widgets in this theme.
|
|
* @param array $instance An array of settings for this widget instance.
|
|
*/
|
|
public function widget( $args, $instance ) {
|
|
|
|
// Merge with defaults.
|
|
$instance = wp_parse_args( (array) $instance, $this->defaults );
|
|
$args = wp_parse_args(
|
|
$args,
|
|
[
|
|
'before_widget' => '',
|
|
'after_widget' => '',
|
|
'before_title' => '',
|
|
'after_title' => '',
|
|
]
|
|
);
|
|
|
|
// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
|
|
echo $args['before_widget'];
|
|
|
|
if ( ! empty( $instance['title'] ) ) {
|
|
// phpcs:ignore
|
|
echo $args['before_title'] . apply_filters( 'widget_title', $instance['title'], $instance, $this->id_base ) . $args['after_title'];
|
|
}
|
|
|
|
if ( ! empty( $instance['form_id'] ) ) {
|
|
wpforms()->obj( 'frontend' )->output( absint( $instance['form_id'] ), (bool) $instance['show_title'], (bool) $instance['show_desc'] );
|
|
}
|
|
|
|
// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
|
|
echo $args['after_widget'];
|
|
}
|
|
|
|
/**
|
|
* Deal with the settings when they are saved by the admin. Here is
|
|
* where any validation should be dealt with.
|
|
*
|
|
* @since 1.0.2
|
|
*
|
|
* @param array $new_instance An array of new settings as submitted by the admin.
|
|
* @param array $old_instance An array of the previous settings.
|
|
*
|
|
* @return array The validated and (if necessary) amended settings
|
|
*/
|
|
public function update( $new_instance, $old_instance ) {
|
|
|
|
$new_instance['title'] = wp_strip_all_tags( $new_instance['title'] );
|
|
$new_instance['form_id'] = ! empty( $new_instance['form_id'] ) ? (int) $new_instance['form_id'] : 0;
|
|
$new_instance['show_title'] = isset( $new_instance['show_title'] ) && $new_instance['show_title'] ? '1' : false;
|
|
$new_instance['show_desc'] = isset( $new_instance['show_desc'] ) && $new_instance['show_desc'] ? '1' : false;
|
|
|
|
return $new_instance;
|
|
}
|
|
|
|
/**
|
|
* Display the form for this widget on the Widgets page of the WP Admin area.
|
|
*
|
|
* @since 1.0.2
|
|
*
|
|
* @param array $instance An array of the current settings for this widget.
|
|
*/
|
|
public function form( $instance ) {
|
|
|
|
// Merge with defaults.
|
|
$instance = wp_parse_args( (array) $instance, $this->defaults );
|
|
?>
|
|
<p>
|
|
<label for="<?php echo esc_attr( $this->get_field_id( 'title' ) ); ?>">
|
|
<?php echo esc_html( _x( 'Title:', 'Widget', 'wpforms-lite' ) ); ?>
|
|
</label>
|
|
<input type="text" id="<?php echo esc_attr( $this->get_field_id( 'title' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'title' ) ); ?>" value="<?php echo esc_attr( $instance['title'] ); ?>" class="widefat"/>
|
|
</p>
|
|
<p>
|
|
<label for="<?php echo esc_attr( $this->get_field_id( 'form_id' ) ); ?>">
|
|
<?php echo esc_html( _x( 'Form:', 'Widget', 'wpforms-lite' ) ); ?>
|
|
</label>
|
|
<select class="widefat" id="<?php echo esc_attr( $this->get_field_id( 'form_id' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'form_id' ) ); ?>">
|
|
<?php
|
|
$forms = wpforms()->obj( 'form' )->get();
|
|
|
|
if ( ! empty( $forms ) ) {
|
|
echo '<option value="" selected disabled>' . esc_html_x( 'Select your form', 'Widget', 'wpforms-lite' ) . '</option>';
|
|
|
|
foreach ( $forms as $form ) {
|
|
echo '<option value="' . esc_attr( $form->ID ) . '" ' . selected( $instance['form_id'], $form->ID, false ) . '>' . esc_html( $form->post_title ) . '</option>';
|
|
}
|
|
} else {
|
|
echo '<option value="">' . esc_html_x( 'No forms', 'Widget', 'wpforms-lite' ) . '</option>';
|
|
}
|
|
?>
|
|
</select>
|
|
</p>
|
|
<p>
|
|
<input type="checkbox" id="<?php echo esc_attr( $this->get_field_id( 'show_title' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'show_title' ) ); ?>" <?php checked( '1', $instance['show_title'] ); ?>>
|
|
<label for="<?php echo esc_attr( $this->get_field_id( 'show_title' ) ); ?>">
|
|
<?php echo esc_html( _x( 'Display form name', 'Widget', 'wpforms-lite' ) ); ?>
|
|
</label>
|
|
<br>
|
|
<input type="checkbox" id="<?php echo esc_attr( $this->get_field_id( 'show_desc' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'show_desc' ) ); ?>" <?php checked( '1', $instance['show_desc'] ); ?>>
|
|
<label for="<?php echo esc_attr( $this->get_field_id( 'show_desc' ) ); ?>">
|
|
<?php echo esc_html( _x( 'Display form description', 'Widget', 'wpforms-lite' ) ); ?>
|
|
</label>
|
|
</p>
|
|
<?php
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Register the WPForms widget.
|
|
*
|
|
* @since 1.0.2
|
|
*/
|
|
function wpforms_register_widgets() { // phpcs:ignore WPForms.PHP.HooksMethod.InvalidPlaceForAddingHooks
|
|
|
|
// Hide legacy widget. This filter is available from WP 5.8.
|
|
// phpcs:disable WPForms.PHP.ValidateHooks.InvalidHookName, WPForms.Comments.PHPDocHooks.RequiredHookDocumentation
|
|
add_filter(
|
|
'widget_types_to_hide_from_legacy_widget_block',
|
|
static function ( $widget_types ) {
|
|
|
|
$widget_types[] = 'wpforms-widget';
|
|
|
|
return $widget_types;
|
|
}
|
|
);
|
|
// phpcs:enable WPForms.PHP.ValidateHooks.InvalidHookName, WPForms.Comments.PHPDocHooks.RequiredHookDocumentation
|
|
|
|
register_widget( 'WPForms_Widget' );
|
|
}
|
|
|
|
add_action( 'widgets_init', 'wpforms_register_widgets' );
|