| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656 |
- <?php
- /**
- ** A base module for [captchac] and [captchar]
- **/
- /* form_tag handler */
- add_action( 'wpcf7_init', 'wpcf7_add_form_tag_captcha', 10, 0 );
- function wpcf7_add_form_tag_captcha() {
- // CAPTCHA-Challenge (image)
- wpcf7_add_form_tag( 'captchac',
- 'wpcf7_captchac_form_tag_handler',
- array(
- 'name-attr' => true,
- 'zero-controls-container' => true,
- 'not-for-mail' => true,
- )
- );
- // CAPTCHA-Response (input)
- wpcf7_add_form_tag( 'captchar',
- 'wpcf7_captchar_form_tag_handler',
- array(
- 'name-attr' => true,
- 'do-not-store' => true,
- 'not-for-mail' => true,
- )
- );
- }
- function wpcf7_captchac_form_tag_handler( $tag ) {
- if ( ! class_exists( 'ReallySimpleCaptcha' ) ) {
- $error = sprintf(
- /* translators: %s: link labeled 'Really Simple CAPTCHA' */
- esc_html( __( "To use CAPTCHA, you need %s plugin installed.", 'contact-form-7' ) ),
- wpcf7_link( 'https://wordpress.org/plugins/really-simple-captcha/', 'Really Simple CAPTCHA' )
- );
- return sprintf( '<em>%s</em>', $error );
- }
- if ( empty( $tag->name ) ) {
- return '';
- }
- $class = wpcf7_form_controls_class( $tag->type );
- $class .= ' wpcf7-captcha-' . $tag->name;
- $atts = array();
- $atts['class'] = $tag->get_class_option( $class );
- $atts['id'] = $tag->get_id_option();
- $op = array( // Default
- 'img_size' => array( 72, 24 ),
- 'base' => array( 6, 18 ),
- 'font_size' => 14,
- 'font_char_width' => 15,
- );
- $op = array_merge( $op, wpcf7_captchac_options( $tag->options ) );
- if ( ! $filename = wpcf7_generate_captcha( $op ) ) {
- return '';
- }
- if ( ! empty( $op['img_size'] ) ) {
- if ( isset( $op['img_size'][0] ) ) {
- $atts['width'] = $op['img_size'][0];
- }
- if ( isset( $op['img_size'][1] ) ) {
- $atts['height'] = $op['img_size'][1];
- }
- }
- $atts['alt'] = 'captcha';
- $atts['src'] = wpcf7_captcha_url( $filename );
- $atts = wpcf7_format_atts( $atts );
- $prefix = substr( $filename, 0, strrpos( $filename, '.' ) );
- $html = sprintf(
- '<input type="hidden" name="_wpcf7_captcha_challenge_%1$s" value="%2$s" /><img %3$s />',
- $tag->name, esc_attr( $prefix ), $atts );
- return $html;
- }
- function wpcf7_captchar_form_tag_handler( $tag ) {
- if ( empty( $tag->name ) ) {
- return '';
- }
- $validation_error = wpcf7_get_validation_error( $tag->name );
- $class = wpcf7_form_controls_class( $tag->type );
- if ( $validation_error ) {
- $class .= ' wpcf7-not-valid';
- }
- $atts = array();
- $atts['size'] = $tag->get_size_option( '40' );
- $atts['maxlength'] = $tag->get_maxlength_option();
- $atts['minlength'] = $tag->get_minlength_option();
- if ( $atts['maxlength'] and $atts['minlength']
- and $atts['maxlength'] < $atts['minlength'] ) {
- unset( $atts['maxlength'], $atts['minlength'] );
- }
- $atts['class'] = $tag->get_class_option( $class );
- $atts['id'] = $tag->get_id_option();
- $atts['tabindex'] = $tag->get_option( 'tabindex', 'signed_int', true );
- $atts['autocomplete'] = 'off';
- $atts['aria-invalid'] = $validation_error ? 'true' : 'false';
- $value = (string) reset( $tag->values );
- if ( wpcf7_is_posted() ) {
- $value = '';
- }
- if ( $tag->has_option( 'placeholder' )
- or $tag->has_option( 'watermark' ) ) {
- $atts['placeholder'] = $value;
- $value = '';
- }
- $atts['value'] = $value;
- $atts['type'] = 'text';
- $atts['name'] = $tag->name;
- $atts = wpcf7_format_atts( $atts );
- $html = sprintf(
- '<span class="wpcf7-form-control-wrap %1$s"><input %2$s />%3$s</span>',
- sanitize_html_class( $tag->name ), $atts, $validation_error );
- return $html;
- }
- /* Validation filter */
- add_filter( 'wpcf7_validate_captchar',
- 'wpcf7_captcha_validation_filter', 10, 2 );
- function wpcf7_captcha_validation_filter( $result, $tag ) {
- $type = $tag->type;
- $name = $tag->name;
- $captchac = '_wpcf7_captcha_challenge_' . $name;
- $prefix = isset( $_POST[$captchac] ) ? (string) $_POST[$captchac] : '';
- $response = isset( $_POST[$name] ) ? (string) $_POST[$name] : '';
- $response = wpcf7_canonicalize( $response );
- if ( 0 == strlen( $prefix )
- or ! wpcf7_check_captcha( $prefix, $response ) ) {
- $result->invalidate( $tag, wpcf7_get_message( 'captcha_not_match' ) );
- }
- if ( 0 != strlen( $prefix ) ) {
- wpcf7_remove_captcha( $prefix );
- }
- return $result;
- }
- /* Ajax echo filter */
- add_filter( 'wpcf7_ajax_onload', 'wpcf7_captcha_ajax_refill', 10, 1 );
- add_filter( 'wpcf7_ajax_json_echo', 'wpcf7_captcha_ajax_refill', 10, 1 );
- function wpcf7_captcha_ajax_refill( $items ) {
- if ( ! is_array( $items ) ) {
- return $items;
- }
- $tags = wpcf7_scan_form_tags( array( 'type' => 'captchac' ) );
- if ( empty( $tags ) ) {
- return $items;
- }
- $refill = array();
- foreach ( $tags as $tag ) {
- $name = $tag->name;
- $options = $tag->options;
- if ( empty( $name ) ) {
- continue;
- }
- $op = wpcf7_captchac_options( $options );
- if ( $filename = wpcf7_generate_captcha( $op ) ) {
- $captcha_url = wpcf7_captcha_url( $filename );
- $refill[$name] = $captcha_url;
- }
- }
- if ( ! empty( $refill ) ) {
- $items['captcha'] = $refill;
- }
- return $items;
- }
- /* Messages */
- add_filter( 'wpcf7_messages', 'wpcf7_captcha_messages', 10, 1 );
- function wpcf7_captcha_messages( $messages ) {
- $messages = array_merge( $messages, array(
- 'captcha_not_match' => array(
- 'description' =>
- __( "The code that sender entered does not match the CAPTCHA", 'contact-form-7' ),
- 'default' =>
- __( 'Your entered code is incorrect.', 'contact-form-7' ),
- ),
- ) );
- return $messages;
- }
- /* Tag generator */
- add_action( 'wpcf7_admin_init', 'wpcf7_add_tag_generator_captcha', 46, 0 );
- function wpcf7_add_tag_generator_captcha() {
- if ( ! wpcf7_use_really_simple_captcha() ) {
- return;
- }
- $tag_generator = WPCF7_TagGenerator::get_instance();
- $tag_generator->add( 'captcha',
- __( 'CAPTCHA (Really Simple CAPTCHA)', 'contact-form-7' ),
- 'wpcf7_tag_generator_captcha' );
- }
- function wpcf7_tag_generator_captcha( $contact_form, $args = '' ) {
- $args = wp_parse_args( $args, array() );
- if ( ! class_exists( 'ReallySimpleCaptcha' ) ) {
- ?>
- <div class="control-box">
- <fieldset>
- <legend><?php
- echo sprintf(
- /* translators: %s: link labeled 'Really Simple CAPTCHA' */
- esc_html( __( "To use CAPTCHA, you first need to install and activate %s plugin.", 'contact-form-7' ) ),
- wpcf7_link( 'https://wordpress.org/plugins/really-simple-captcha/', 'Really Simple CAPTCHA' )
- );
- ?></legend>
- </fieldset>
- </div>
- <?php
- return;
- }
- $description = __( "Generate form-tags for a CAPTCHA image and corresponding response input field. For more details, see %s.", 'contact-form-7' );
- $desc_link = wpcf7_link( __( 'https://contactform7.com/captcha/', 'contact-form-7' ), __( 'CAPTCHA', 'contact-form-7' ) );
- ?>
- <div class="control-box">
- <fieldset>
- <legend><?php echo sprintf( esc_html( $description ), $desc_link ); ?></legend>
- <table class="form-table">
- <tbody>
- <tr>
- <th scope="row"><label for="<?php echo esc_attr( $args['content'] . '-name' ); ?>"><?php echo esc_html( __( 'Name', 'contact-form-7' ) ); ?></label></th>
- <td><input type="text" name="name" class="tg-name oneline" id="<?php echo esc_attr( $args['content'] . '-name' ); ?>" /></td>
- </tr>
- </tbody>
- </table>
- <table class="form-table scope captchac">
- <caption><?php echo esc_html( __( "Image settings", 'contact-form-7' ) ); ?></caption>
- <tbody>
- <tr>
- <th scope="row"><label for="<?php echo esc_attr( $args['content'] . '-captchac-id' ); ?>"><?php echo esc_html( __( 'Id attribute', 'contact-form-7' ) ); ?></label></th>
- <td><input type="text" name="id" class="idvalue oneline option" id="<?php echo esc_attr( $args['content'] . '-captchac-id' ); ?>" /></td>
- </tr>
- <tr>
- <th scope="row"><label for="<?php echo esc_attr( $args['content'] . '-captchac-class' ); ?>"><?php echo esc_html( __( 'Class attribute', 'contact-form-7' ) ); ?></label></th>
- <td><input type="text" name="class" class="classvalue oneline option" id="<?php echo esc_attr( $args['content'] . '-captchac-class' ); ?>" /></td>
- </tr>
- </tbody>
- </table>
- <table class="form-table scope captchar">
- <caption><?php echo esc_html( __( "Input field settings", 'contact-form-7' ) ); ?></caption>
- <tbody>
- <tr>
- <th scope="row"><label for="<?php echo esc_attr( $args['content'] . '-captchar-id' ); ?>"><?php echo esc_html( __( 'Id attribute', 'contact-form-7' ) ); ?></label></th>
- <td><input type="text" name="id" class="idvalue oneline option" id="<?php echo esc_attr( $args['content'] . '-captchar-id' ); ?>" /></td>
- </tr>
- <tr>
- <th scope="row"><label for="<?php echo esc_attr( $args['content'] . '-captchar-class' ); ?>"><?php echo esc_html( __( 'Class attribute', 'contact-form-7' ) ); ?></label></th>
- <td><input type="text" name="class" class="classvalue oneline option" id="<?php echo esc_attr( $args['content'] . '-captchar-class' ); ?>" /></td>
- </tr>
- </tbody>
- </table>
- </fieldset>
- </div>
- <div class="insert-box">
- <input type="text" name="captcha" class="tag code" readonly="readonly" onfocus="this.select()" />
- <div class="submitbox">
- <input type="button" class="button button-primary insert-tag" value="<?php echo esc_attr( __( 'Insert Tag', 'contact-form-7' ) ); ?>" />
- </div>
- </div>
- <?php
- }
- /* Warning message */
- add_action( 'wpcf7_admin_warnings',
- 'wpcf7_captcha_display_warning_message', 10, 3 );
- function wpcf7_captcha_display_warning_message( $page, $action, $object ) {
- if ( $object instanceof WPCF7_ContactForm ) {
- $contact_form = $object;
- } else {
- return;
- }
- $has_tags = (bool) $contact_form->scan_form_tags(
- array( 'type' => array( 'captchac' ) ) );
- if ( ! $has_tags ) {
- return;
- }
- if ( ! class_exists( 'ReallySimpleCaptcha' ) ) {
- return;
- }
- $uploads_dir = wpcf7_captcha_tmp_dir();
- wpcf7_init_captcha();
- if ( ! is_dir( $uploads_dir )
- or ! wp_is_writable( $uploads_dir ) ) {
- $message = sprintf( __( 'This contact form contains CAPTCHA fields, but the temporary folder for the files (%s) does not exist or is not writable. You can create the folder or change its permission manually.', 'contact-form-7' ), $uploads_dir );
- echo '<div class="notice notice-warning"><p>' . esc_html( $message ) . '</p></div>';
- }
- if ( ! function_exists( 'imagecreatetruecolor' )
- or ! function_exists( 'imagettftext' ) ) {
- $message = __( "This contact form contains CAPTCHA fields, but the necessary libraries (GD and FreeType) are not available on your server.", 'contact-form-7' );
- echo '<div class="notice notice-warning"><p>' . esc_html( $message ) . '</p></div>';
- }
- }
- /* CAPTCHA functions */
- function wpcf7_init_captcha() {
- static $captcha = null;
- if ( $captcha ) {
- return $captcha;
- }
- if ( class_exists( 'ReallySimpleCaptcha' ) ) {
- $captcha = new ReallySimpleCaptcha();
- } else {
- return false;
- }
- $dir = trailingslashit( wpcf7_captcha_tmp_dir() );
- $captcha->tmp_dir = $dir;
- if ( is_callable( array( $captcha, 'make_tmp_dir' ) ) ) {
- $result = $captcha->make_tmp_dir();
- if ( ! $result ) {
- return false;
- }
- return $captcha;
- }
- if ( wp_mkdir_p( $dir ) ) {
- $htaccess_file = path_join( $dir, '.htaccess' );
- if ( file_exists( $htaccess_file ) ) {
- return $captcha;
- }
- if ( $handle = fopen( $htaccess_file, 'w' ) ) {
- fwrite( $handle, 'Order deny,allow' . "\n" );
- fwrite( $handle, 'Deny from all' . "\n" );
- fwrite( $handle, '<Files ~ "^[0-9A-Za-z]+\\.(jpeg|gif|png)$">' . "\n" );
- fwrite( $handle, ' Allow from all' . "\n" );
- fwrite( $handle, '</Files>' . "\n" );
- fclose( $handle );
- }
- } else {
- return false;
- }
- return $captcha;
- }
- function wpcf7_captcha_tmp_dir() {
- if ( defined( 'WPCF7_CAPTCHA_TMP_DIR' ) ) {
- return WPCF7_CAPTCHA_TMP_DIR;
- } else {
- return path_join( wpcf7_upload_dir( 'dir' ), 'wpcf7_captcha' );
- }
- }
- function wpcf7_captcha_tmp_url() {
- if ( defined( 'WPCF7_CAPTCHA_TMP_URL' ) ) {
- return WPCF7_CAPTCHA_TMP_URL;
- } else {
- return path_join( wpcf7_upload_dir( 'url' ), 'wpcf7_captcha' );
- }
- }
- function wpcf7_captcha_url( $filename ) {
- $url = path_join( wpcf7_captcha_tmp_url(), $filename );
- if ( is_ssl()
- and 'http:' == substr( $url, 0, 5 ) ) {
- $url = 'https:' . substr( $url, 5 );
- }
- return apply_filters( 'wpcf7_captcha_url', esc_url_raw( $url ) );
- }
- function wpcf7_generate_captcha( $options = null ) {
- if ( ! $captcha = wpcf7_init_captcha() ) {
- return false;
- }
- if ( ! is_dir( $captcha->tmp_dir )
- or ! wp_is_writable( $captcha->tmp_dir ) ) {
- return false;
- }
- $img_type = imagetypes();
- if ( $img_type & IMG_PNG ) {
- $captcha->img_type = 'png';
- } elseif ( $img_type & IMG_GIF ) {
- $captcha->img_type = 'gif';
- } elseif ( $img_type & IMG_JPG ) {
- $captcha->img_type = 'jpeg';
- } else {
- return false;
- }
- if ( is_array( $options ) ) {
- if ( isset( $options['img_size'] ) ) {
- $captcha->img_size = $options['img_size'];
- }
- if ( isset( $options['base'] ) ) {
- $captcha->base = $options['base'];
- }
- if ( isset( $options['font_size'] ) ) {
- $captcha->font_size = $options['font_size'];
- }
- if ( isset( $options['font_char_width'] ) ) {
- $captcha->font_char_width = $options['font_char_width'];
- }
- if ( isset( $options['fg'] ) ) {
- $captcha->fg = $options['fg'];
- }
- if ( isset( $options['bg'] ) ) {
- $captcha->bg = $options['bg'];
- }
- }
- $prefix = wp_rand();
- $captcha_word = $captcha->generate_random_word();
- return $captcha->generate_image( $prefix, $captcha_word );
- }
- function wpcf7_check_captcha( $prefix, $response ) {
- if ( ! $captcha = wpcf7_init_captcha() ) {
- return false;
- }
- return $captcha->check( $prefix, $response );
- }
- function wpcf7_remove_captcha( $prefix ) {
- if ( ! $captcha = wpcf7_init_captcha() ) {
- return false;
- }
- // Contact Form 7 generates $prefix with wp_rand()
- if ( preg_match( '/[^0-9]/', $prefix ) ) {
- return false;
- }
- $captcha->remove( $prefix );
- }
- add_action( 'template_redirect', 'wpcf7_cleanup_captcha_files', 20, 0 );
- function wpcf7_cleanup_captcha_files() {
- if ( ! $captcha = wpcf7_init_captcha() ) {
- return false;
- }
- if ( is_callable( array( $captcha, 'cleanup' ) ) ) {
- return $captcha->cleanup();
- }
- $dir = trailingslashit( wpcf7_captcha_tmp_dir() );
- if ( ! is_dir( $dir )
- or ! is_readable( $dir )
- or ! wp_is_writable( $dir ) ) {
- return false;
- }
- if ( $handle = opendir( $dir ) ) {
- while ( false !== ( $file = readdir( $handle ) ) ) {
- if ( ! preg_match( '/^[0-9]+\.(php|txt|png|gif|jpeg)$/', $file ) ) {
- continue;
- }
- $stat = stat( path_join( $dir, $file ) );
- if ( $stat['mtime'] + 3600 < time() ) { // 3600 secs == 1 hour
- @unlink( path_join( $dir, $file ) );
- }
- }
- closedir( $handle );
- }
- }
- function wpcf7_captchac_options( $options ) {
- if ( ! is_array( $options ) ) {
- return array();
- }
- $op = array();
- $image_size_array = preg_grep( '%^size:[smlSML]$%', $options );
- if ( $image_size = array_shift( $image_size_array ) ) {
- preg_match( '%^size:([smlSML])$%', $image_size, $is_matches );
- switch ( strtolower( $is_matches[1] ) ) {
- case 's':
- $op['img_size'] = array( 60, 20 );
- $op['base'] = array( 6, 15 );
- $op['font_size'] = 11;
- $op['font_char_width'] = 13;
- break;
- case 'l':
- $op['img_size'] = array( 84, 28 );
- $op['base'] = array( 6, 20 );
- $op['font_size'] = 17;
- $op['font_char_width'] = 19;
- break;
- case 'm':
- default:
- $op['img_size'] = array( 72, 24 );
- $op['base'] = array( 6, 18 );
- $op['font_size'] = 14;
- $op['font_char_width'] = 15;
- }
- }
- $fg_color_array = preg_grep(
- '%^fg:#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$%', $options );
- if ( $fg_color = array_shift( $fg_color_array ) ) {
- preg_match( '%^fg:#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$%',
- $fg_color, $fc_matches );
- if ( 3 == strlen( $fc_matches[1] ) ) {
- $r = substr( $fc_matches[1], 0, 1 );
- $g = substr( $fc_matches[1], 1, 1 );
- $b = substr( $fc_matches[1], 2, 1 );
- $op['fg'] = array(
- hexdec( $r . $r ),
- hexdec( $g . $g ),
- hexdec( $b . $b ),
- );
- } elseif ( 6 == strlen( $fc_matches[1] ) ) {
- $r = substr( $fc_matches[1], 0, 2 );
- $g = substr( $fc_matches[1], 2, 2 );
- $b = substr( $fc_matches[1], 4, 2 );
- $op['fg'] = array(
- hexdec( $r ),
- hexdec( $g ),
- hexdec( $b ),
- );
- }
- }
- $bg_color_array = preg_grep(
- '%^bg:#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$%', $options );
- if ( $bg_color = array_shift( $bg_color_array ) ) {
- preg_match( '%^bg:#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$%',
- $bg_color, $bc_matches );
- if ( 3 == strlen( $bc_matches[1] ) ) {
- $r = substr( $bc_matches[1], 0, 1 );
- $g = substr( $bc_matches[1], 1, 1 );
- $b = substr( $bc_matches[1], 2, 1 );
- $op['bg'] = array(
- hexdec( $r . $r ),
- hexdec( $g . $g ),
- hexdec( $b . $b ),
- );
- } elseif ( 6 == strlen( $bc_matches[1] ) ) {
- $r = substr( $bc_matches[1], 0, 2 );
- $g = substr( $bc_matches[1], 2, 2 );
- $b = substr( $bc_matches[1], 4, 2 );
- $op['bg'] = array(
- hexdec( $r ),
- hexdec( $g ),
- hexdec( $b ),
- );
- }
- }
- return $op;
- }
|