diff options
Diffstat (limited to 'plugins/jetpack/modules/contact-form')
37 files changed, 9616 insertions, 0 deletions
diff --git a/plugins/jetpack/modules/contact-form/admin.php b/plugins/jetpack/modules/contact-form/admin.php new file mode 100644 index 00000000..0596f798 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/admin.php @@ -0,0 +1,894 @@ +<?php +/** + * Add a contact form button to the post composition screen + */ +add_action( 'media_buttons', 'grunion_media_button', 999 ); +function grunion_media_button() { + global $post_ID, $temp_ID, $pagenow; + + if ( 'press-this.php' === $pagenow ) { + return; + } + + $iframe_post_id = (int) ( 0 == $post_ID ? $temp_ID : $post_ID ); + $title = __( 'Add Contact Form', 'jetpack' ); + $plugin_url = esc_url( GRUNION_PLUGIN_URL ); + $site_url = esc_url( admin_url( "/admin-ajax.php?post_id={$iframe_post_id}&action=grunion_form_builder&TB_iframe=true&width=768" ) ); + ?> + + <a id="insert-jetpack-contact-form" class="button thickbox" title="<?php echo esc_attr( $title ); ?>" data-editor="content" href="<?php echo $site_url; ?>&id=add_form"> + <span class="jetpack-contact-form-icon"></span> <?php echo esc_html( $title ); ?> + </a> + + <?php +} + +add_action( 'wp_ajax_grunion_form_builder', 'grunion_display_form_view' ); + +function grunion_display_form_view() { + if ( current_user_can( 'edit_posts' ) ) { + require_once GRUNION_PLUGIN_DIR . 'grunion-form-view.php'; + } + exit; +} + +// feedback specific css items +add_action( 'admin_print_styles', 'grunion_admin_css' ); +function grunion_admin_css() { + global $current_screen; + if ( is_null( $current_screen ) ) { + return; + } + if ( 'edit-feedback' !== $current_screen->id ) { + return; + } + + wp_enqueue_script( 'wp-lists' ); +?> + +<style type='text/css'> +.add-new-h2, .view-switch, body.no-js .tablenav select[name^=action], body.no-js #doaction, body.no-js #doaction2 { + display: none +} + +.column-feedback_from img { + float:left; + margin-right:10px; + margin-top:3px; +} + +.widefat .column-feedback_from { + width: 17%; +} +.widefat .column-feedback_date { + width: 17%; +} + +.spam a { + color: #BC0B0B; +} + +.untrash a { + color: #D98500; +} + +.unspam a { +color: #D98500; +} + +</style> + +<?php +} + +/** + * Hack a 'Bulk Spam' option for bulk edit in other than spam view + * Hack a 'Bulk Delete' option for bulk edit in spam view + * + * There isn't a better way to do this until + * http://core.trac.wordpress.org/changeset/17297 is resolved + */ +add_action( 'admin_head', 'grunion_add_bulk_edit_option' ); +function grunion_add_bulk_edit_option() { + + $screen = get_current_screen(); + + if ( is_null( $screen ) ) { + return; + } + + if ( 'edit-feedback' != $screen->id ) { + return; + } + + // When viewing spam we want to be able to be able to bulk delete + // When viewing anything we want to be able to bulk move to spam + if ( isset( $_GET['post_status'] ) && 'spam' == $_GET['post_status'] ) { + // Create Delete Permanently bulk item + $option_val = 'delete'; + $option_txt = __( 'Delete Permanently', 'jetpack' ); + $pseudo_selector = 'last-child'; + + } else { + // Create Mark Spam bulk item + $option_val = 'spam'; + $option_txt = __( 'Mark as Spam', 'jetpack' ); + $pseudo_selector = 'first-child'; + } + + ?> + <script type="text/javascript"> + jQuery(document).ready(function($) { + $('#posts-filter .actions select').filter('[name=action], [name=action2]').find('option:<?php echo $pseudo_selector; ?>').after('<option value="<?php echo $option_val; ?>"><?php echo esc_attr( $option_txt ); ?></option>' ); + }) + </script> + <?php +} + +/** + * Hack an 'Empty Spam' button to spam view + * + * Leverages core's delete_all functionality + */ +add_action( 'admin_head', 'grunion_add_empty_spam_button' ); +function grunion_add_empty_spam_button() { + $screen = get_current_screen(); + + if ( is_null( $screen ) ) { + return; + } + + // Only add to feedback, only to spam view + if ( 'edit-feedback' != $screen->id + || empty( $_GET['post_status'] ) + || 'spam' !== $_GET['post_status'] ) { + return; + } + + // Get HTML for the button + $button_html = wp_nonce_field( 'bulk-destroy', '_destroy_nonce', true, false ); + $button_html .= get_submit_button( __( 'Empty Spam', 'jetpack' ), 'apply', 'delete_all', false ); + + // Add the button next to the filter button via js + ?> + <script type="text/javascript"> + jQuery(document).ready(function($) { + $('#posts-filter #post-query-submit').after('<?php echo $button_html; ?>' ); + }) + </script> + <?php +} + +/** + * Handle a bulk spam report + */ +add_action( 'admin_init', 'grunion_handle_bulk_spam' ); +function grunion_handle_bulk_spam() { + global $pagenow; + + if ( 'edit.php' != $pagenow + || ( empty( $_REQUEST['post_type'] ) || 'feedback' != $_REQUEST['post_type'] ) ) { + return; + } + + // Slip in a success message + if ( ! empty( $_REQUEST['message'] ) && 'marked-spam' == $_REQUEST['message'] ) { + add_action( 'admin_notices', 'grunion_message_bulk_spam' ); + } + + if ( ( empty( $_REQUEST['action'] ) || 'spam' != $_REQUEST['action'] ) && ( empty( $_REQUEST['action2'] ) || 'spam' != $_REQUEST['action2'] ) ) { + return; + } + + check_admin_referer( 'bulk-posts' ); + + if ( empty( $_REQUEST['post'] ) ) { + wp_safe_redirect( wp_get_referer() ); + exit; + } + + $post_ids = array_map( 'intval', $_REQUEST['post'] ); + + foreach ( $post_ids as $post_id ) { + if ( ! current_user_can( 'edit_page', $post_id ) ) { + wp_die( __( 'You are not allowed to manage this item.', 'jetpack' ) ); + } + + $post = array( + 'ID' => $post_id, + 'post_status' => 'spam', + ); + $akismet_values = get_post_meta( $post_id, '_feedback_akismet_values', true ); + wp_update_post( $post ); + + /** + * Fires after a comment has been marked by Akismet. + * + * Typically this means the comment is spam. + * + * @module contact-form + * + * @since 2.2.0 + * + * @param string $comment_status Usually is 'spam', otherwise 'ham'. + * @param array $akismet_values From '_feedback_akismet_values' in comment meta + */ + do_action( 'contact_form_akismet', 'spam', $akismet_values ); + } + + $redirect_url = add_query_arg( 'message', 'marked-spam', wp_get_referer() ); + wp_safe_redirect( $redirect_url ); + exit; +} + +function grunion_message_bulk_spam() { + echo '<div class="updated"><p>' . __( 'Feedback(s) marked as spam', 'jetpack' ) . '</p></div>'; +} + +// remove admin UI parts that we don't support in feedback management +add_action( 'admin_menu', 'grunion_admin_menu' ); +function grunion_admin_menu() { + global $menu, $submenu; + unset( $submenu['edit.php?post_type=feedback'] ); +} + +add_filter( 'bulk_actions-edit-feedback', 'grunion_admin_bulk_actions' ); +function grunion_admin_bulk_actions( $actions ) { + global $current_screen; + if ( 'edit-feedback' != $current_screen->id ) { + return $actions; + } + + unset( $actions['edit'] ); + return $actions; +} + +add_filter( 'views_edit-feedback', 'grunion_admin_view_tabs' ); +function grunion_admin_view_tabs( $views ) { + global $current_screen; + if ( 'edit-feedback' != $current_screen->id ) { + return $views; + } + + unset( $views['publish'] ); + + preg_match( '|post_type=feedback\'( class="current")?\>(.*)\<span class=|', $views['all'], $match ); + if ( ! empty( $match[2] ) ) { + $views['all'] = str_replace( $match[2], __( 'Messages', 'jetpack' ) . ' ', $views['all'] ); + } + + return $views; +} + +add_filter( 'manage_feedback_posts_columns', 'grunion_post_type_columns_filter' ); +function grunion_post_type_columns_filter( $cols ) { + $cols = array( + 'cb' => '<input type="checkbox" />', + 'feedback_from' => __( 'From', 'jetpack' ), + 'feedback_message' => __( 'Message', 'jetpack' ), + 'feedback_date' => __( 'Date', 'jetpack' ), + ); + + return $cols; +} + +add_action( 'manage_posts_custom_column', 'grunion_manage_post_columns', 10, 2 ); +function grunion_manage_post_columns( $col, $post_id ) { + global $post; + + /** + * Only call parse_fields_from_content if we're dealing with a Grunion custom column. + */ + if ( ! in_array( $col, array( 'feedback_date', 'feedback_from', 'feedback_message' ) ) ) { + return; + } + + $content_fields = Grunion_Contact_Form_Plugin::parse_fields_from_content( $post_id ); + + switch ( $col ) { + case 'feedback_from': + $author_name = isset( $content_fields['_feedback_author'] ) ? $content_fields['_feedback_author'] : ''; + $author_email = isset( $content_fields['_feedback_author_email'] ) ? $content_fields['_feedback_author_email'] : ''; + $author_url = isset( $content_fields['_feedback_author_url'] ) ? $content_fields['_feedback_author_url'] : ''; + $author_ip = isset( $content_fields['_feedback_ip'] ) ? $content_fields['_feedback_ip'] : ''; + $form_url = isset( $post->post_parent ) ? get_permalink( $post->post_parent ) : null; + + $author_name_line = ''; + if ( ! empty( $author_name ) ) { + if ( ! empty( $author_email ) ) { + $author_name_line = get_avatar( $author_email, 32 ); + } + + $author_name_line .= sprintf( '<strong>%s</strong><br />', esc_html( $author_name ) ); + } + + $author_email_line = ''; + if ( ! empty( $author_email ) ) { + $author_email_line = sprintf( "<a href='%1\$s' target='_blank'>%2\$s</a><br />", esc_url( 'mailto:' . $author_email ), esc_html( $author_email ) ); + } + + $author_url_line = ''; + if ( ! empty( $author_url ) ) { + $author_url_line = sprintf( "<a href='%1\$s'>%1\$s</a><br />", esc_url( $author_url ) ); + } + + echo $author_name_line; + echo $author_email_line; + echo $author_url_line; + echo "<a href='edit.php?post_type=feedback&s=" . urlencode( $author_ip ); + echo "&mode=detail'>" . esc_html( $author_ip ) . '</a><br />'; + if ( $form_url ) { + echo '<a href="' . esc_url( $form_url ) . '">' . esc_html( $form_url ) . '</a>'; + } + break; + + case 'feedback_message': + $post_type_object = get_post_type_object( $post->post_type ); + if ( isset( $content_fields['_feedback_subject'] ) ) { + echo '<strong>'; + echo esc_html( $content_fields['_feedback_subject'] ); + echo '</strong>'; + echo '<br />'; + } + echo sanitize_text_field( get_the_content( '' ) ); + echo '<br />'; + + $extra_fields = get_post_meta( $post_id, '_feedback_extra_fields', true ); + if ( ! empty( $extra_fields ) ) { + echo '<br /><hr />'; + echo '<table cellspacing="0" cellpadding="0" style="">' . "\n"; + foreach ( (array) $extra_fields as $k => $v ) { + // Remove prefix from exta fields + echo "<tr><td align='right'><b>" . esc_html( preg_replace( '#^\d+_#', '', $k ) ) . '</b></td><td>' . sanitize_text_field( $v ) . "</td></tr>\n"; + } + echo '</table>'; + } + + echo '<div class="row-actions">'; + if ( $post->post_status == 'trash' ) { + echo '<span class="untrash" id="feedback-restore-' . $post_id; + echo '"><a title="'; + echo esc_attr__( 'Restore this item from the Trash', 'jetpack' ); + echo '" href="' . wp_nonce_url( admin_url( sprintf( $post_type_object->_edit_link . '&action=untrash', $post->ID ) ), 'untrash-' . $post->post_type . '_' . $post->ID ); + echo '">' . __( 'Restore', 'jetpack' ) . '</a></span> | '; + + echo "<span class='delete'> <a class='submitdelete' title='"; + echo esc_attr( __( 'Delete this item permanently', 'jetpack' ) ); + echo "' href='" . get_delete_post_link( $post->ID, '', true ); + echo "'>" . __( 'Delete Permanently', 'jetpack' ) . '</a></span>'; +?> + +<script> +jQuery(document).ready(function($) { +$('#feedback-restore-<?php echo $post_id; ?>').click(function(e) { + e.preventDefault(); + $.post(ajaxurl, { + action: 'grunion_ajax_spam', + post_id: '<?php echo $post_id; ?>', + make_it: 'publish', + sub_menu: jQuery('.subsubsub .current').attr('href'), + _ajax_nonce: '<?php echo wp_create_nonce( 'grunion-post-status-' . $post_id ); ?>' + }, + function(r) { + $('#post-<?php echo $post_id; ?>') + .css({backgroundColor: '#59C859'}) + .fadeOut(350, function() { + $(this).remove(); + $('.subsubsub').html(r); + }); + } + ); +}); +}); +</script> + +<?php + } elseif ( $post->post_status == 'publish' ) { + echo '<span class="spam" id="feedback-spam-' . $post_id; + echo '"><a title="'; + echo __( 'Mark this message as spam', 'jetpack' ); + echo '" href="' . wp_nonce_url( admin_url( 'admin-ajax.php?post_id=' . $post_id . '&action=spam' ), 'spam-feedback_' . $post_id ); + echo '">Spam</a></span>'; + echo ' | '; + + echo '<span class="delete" id="feedback-trash-' . $post_id; + echo '">'; + echo '<a class="submitdelete" title="' . esc_attr__( 'Trash', 'jetpack' ); + echo '" href="' . get_delete_post_link( $post_id ); + echo '">' . __( 'Trash', 'jetpack' ) . '</a></span>'; + +?> + +<script> +jQuery(document).ready( function($) { + $('#feedback-spam-<?php echo $post_id; ?>').click( function(e) { + e.preventDefault(); + $.post( ajaxurl, { + action: 'grunion_ajax_spam', + post_id: '<?php echo $post_id; ?>', + make_it: 'spam', + sub_menu: jQuery('.subsubsub .current').attr('href'), + _ajax_nonce: '<?php echo wp_create_nonce( 'grunion-post-status-' . $post_id ); ?>' + }, + function( r ) { + $('#post-<?php echo $post_id; ?>') + .css( {backgroundColor:'#FF7979'} ) + .fadeOut(350, function() { + $(this).remove(); + $('.subsubsub').html(r); + }); + }); + }); + + $('#feedback-trash-<?php echo $post_id; ?>').click(function(e) { + e.preventDefault(); + $.post(ajaxurl, { + action: 'grunion_ajax_spam', + post_id: '<?php echo $post_id; ?>', + make_it: 'trash', + sub_menu: jQuery('.subsubsub .current').attr('href'), + _ajax_nonce: '<?php echo wp_create_nonce( 'grunion-post-status-' . $post_id ); ?>' + }, + function(r) { + $('#post-<?php echo $post_id; ?>') + .css({backgroundColor: '#FF7979'}) + .fadeOut(350, function() { + $(this).remove(); + $('.subsubsub').html(r); + }); + } + ); + }); +}); +</script> + +<?php + } elseif ( $post->post_status == 'spam' ) { + echo '<span class="unspam unapprove" id="feedback-ham-' . $post_id; + echo '"><a title="'; + echo __( 'Mark this message as NOT spam', 'jetpack' ); + echo '" href="">Not Spam</a></span>'; + echo ' | '; + + echo "<span class='delete' id='feedback-trash-" . $post_id; + echo "'> <a class='submitdelete' title='"; + echo esc_attr( __( 'Delete this item permanently', 'jetpack' ) ); + echo "' href='" . get_delete_post_link( $post->ID, '', true ); + echo "'>" . __( 'Delete Permanently', 'jetpack' ) . '</a></span>'; +?> + +<script> +jQuery(document).ready( function($) { + $('#feedback-ham-<?php echo $post_id; ?>').click( function(e) { + e.preventDefault(); + $.post( ajaxurl, { + action: 'grunion_ajax_spam', + post_id: '<?php echo $post_id; ?>', + make_it: 'ham', + sub_menu: jQuery('.subsubsub .current').attr('href'), + _ajax_nonce: '<?php echo wp_create_nonce( 'grunion-post-status-' . $post_id ); ?>' + }, + function( r ) { + $('#post-<?php echo $post_id; ?>') + .css( {backgroundColor:'#59C859'} ) + .fadeOut(350, function() { + $(this).remove(); + $('.subsubsub').html(r); + }); + }); + }); +}); +</script> + +<?php + } + break; + + case 'feedback_date': + $date_time_format = _x( '%1$s \a\t %2$s', '{$date_format} \a\t {$time_format}', 'jetpack' ); + $date_time_format = sprintf( $date_time_format, get_option( 'date_format' ), get_option( 'time_format' ) ); + $time = date_i18n( $date_time_format, get_the_time( 'U' ) ); + + echo $time; + break; + } +} + +function grunion_esc_attr( $attr ) { + $out = esc_attr( $attr ); + // we also have to entity-encode square brackets so they don't interfere with the shortcode parser + // FIXME: do this better - just stripping out square brackets for now since they mysteriously keep reappearing + $out = str_replace( '[', '', $out ); + $out = str_replace( ']', '', $out ); + return $out; +} + +function grunion_sort_objects( $a, $b ) { + if ( isset( $a['order'] ) && isset( $b['order'] ) ) { + return $a['order'] - $b['order']; + } + return 0; +} + +// take an array of field types from the form builder, and construct a shortcode form +// returns both the shortcode form, and HTML markup representing a preview of the form +function grunion_ajax_shortcode() { + check_ajax_referer( 'grunion_shortcode' ); + + if ( ! current_user_can( 'edit_posts' ) ) { + die( '-1' ); + } + + $attributes = array(); + + foreach ( array( 'subject', 'to' ) as $attribute ) { + if ( isset( $_POST[ $attribute ] ) && strlen( $_POST[ $attribute ] ) ) { + $attributes[ $attribute ] = stripslashes( $_POST[ $attribute ] ); + } + } + + if ( is_array( $_POST['fields'] ) ) { + $fields = stripslashes_deep( $_POST['fields'] ); + usort( $fields, 'grunion_sort_objects' ); + + $field_shortcodes = array(); + + foreach ( $fields as $field ) { + $field_attributes = array(); + + if ( isset( $field['required'] ) && 'true' === $field['required'] ) { + $field_attributes['required'] = 'true'; + } + + foreach ( array( 'options', 'label', 'type' ) as $attribute ) { + if ( isset( $field[ $attribute ] ) ) { + $field_attributes[ $attribute ] = $field[ $attribute ]; + } + } + + $field_shortcodes[] = new Grunion_Contact_Form_Field( $field_attributes ); + } + } + + $grunion = new Grunion_Contact_Form( $attributes, $field_shortcodes ); + + die( "\n$grunion\n" ); +} + +// takes a post_id, extracts the contact-form shortcode from that post (if there is one), parses it, +// and constructs a json object representing its contents and attributes +function grunion_ajax_shortcode_to_json() { + global $post, $grunion_form; + + check_ajax_referer( 'grunion_shortcode_to_json' ); + + if ( ! empty( $_POST['post_id'] ) && ! current_user_can( 'edit_post', $_POST['post_id'] ) ) { + die( '-1' ); + } elseif ( ! current_user_can( 'edit_posts' ) ) { + die( '-1' ); + } + + if ( ! isset( $_POST['content'] ) || ! is_numeric( $_POST['post_id'] ) ) { + die( '-1' ); + } + + $content = stripslashes( $_POST['content'] ); + + // doesn't look like a post with a [contact-form] already. + if ( false === has_shortcode( $content, 'contact-form' ) ) { + die( '' ); + } + + $post = get_post( $_POST['post_id'] ); + + do_shortcode( $content ); + + $grunion = Grunion_Contact_Form::$last; + + $out = array( + 'to' => '', + 'subject' => '', + 'fields' => array(), + ); + + foreach ( $grunion->fields as $field ) { + $out['fields'][ $field->get_attribute( 'id' ) ] = $field->attributes; + } + + $to = $grunion->get_attribute( 'to' ); + $subject = $grunion->get_attribute( 'subject' ); + foreach ( array( 'to', 'subject' ) as $attribute ) { + $value = $grunion->get_attribute( $attribute ); + if ( isset( $grunion->defaults[ $attribute ] ) && $value == $grunion->defaults[ $attribute ] ) { + $value = ''; + } + $out[ $attribute ] = $value; + } + + die( json_encode( $out ) ); +} + + +add_action( 'wp_ajax_grunion_shortcode', 'grunion_ajax_shortcode' ); +add_action( 'wp_ajax_grunion_shortcode_to_json', 'grunion_ajax_shortcode_to_json' ); + + +// process row-action spam/not spam clicks +add_action( 'wp_ajax_grunion_ajax_spam', 'grunion_ajax_spam' ); +function grunion_ajax_spam() { + global $wpdb; + + if ( empty( $_POST['make_it'] ) ) { + return; + } + + $post_id = (int) $_POST['post_id']; + check_ajax_referer( 'grunion-post-status-' . $post_id ); + if ( ! current_user_can( 'edit_page', $post_id ) ) { + wp_die( __( 'You are not allowed to manage this item.', 'jetpack' ) ); + } + + require_once dirname( __FILE__ ) . '/grunion-contact-form.php'; + + $current_menu = ''; + if ( isset( $_POST['sub_menu'] ) && preg_match( '|post_type=feedback|', $_POST['sub_menu'] ) ) { + if ( preg_match( '|post_status=spam|', $_POST['sub_menu'] ) ) { + $current_menu = 'spam'; + } elseif ( preg_match( '|post_status=trash|', $_POST['sub_menu'] ) ) { + $current_menu = 'trash'; + } else { + $current_menu = 'messages'; + } + } + + $post = get_post( $post_id ); + $post_type_object = get_post_type_object( $post->post_type ); + $akismet_values = get_post_meta( $post_id, '_feedback_akismet_values', true ); + if ( $_POST['make_it'] == 'spam' ) { + $post->post_status = 'spam'; + $status = wp_insert_post( $post ); + wp_transition_post_status( 'spam', 'publish', $post ); + + /** This action is already documented in modules/contact-form/admin.php */ + do_action( 'contact_form_akismet', 'spam', $akismet_values ); + } elseif ( $_POST['make_it'] == 'ham' ) { + $post->post_status = 'publish'; + $status = wp_insert_post( $post ); + wp_transition_post_status( 'publish', 'spam', $post ); + + /** This action is already documented in modules/contact-form/admin.php */ + do_action( 'contact_form_akismet', 'ham', $akismet_values ); + + $comment_author_email = $reply_to_addr = $message = $to = $headers = false; + $blog_url = wp_parse_url( site_url() ); + + // resend the original email + $email = get_post_meta( $post_id, '_feedback_email', true ); + $content_fields = Grunion_Contact_Form_Plugin::parse_fields_from_content( $post_id ); + + if ( ! empty( $email ) && ! empty( $content_fields ) ) { + if ( isset( $content_fields['_feedback_author_email'] ) ) { + $comment_author_email = $content_fields['_feedback_author_email']; + } + + if ( isset( $email['to'] ) ) { + $to = $email['to']; + } + + if ( isset( $email['message'] ) ) { + $message = $email['message']; + } + + if ( isset( $email['headers'] ) ) { + $headers = $email['headers']; + } else { + $headers = 'From: "' . $content_fields['_feedback_author'] . '" <wordpress@' . $blog_url['host'] . ">\r\n"; + + if ( ! empty( $comment_author_email ) ) { + $reply_to_addr = $comment_author_email; + } elseif ( is_array( $to ) ) { + $reply_to_addr = $to[0]; + } + + if ( $reply_to_addr ) { + $headers .= 'Reply-To: "' . $content_fields['_feedback_author'] . '" <' . $reply_to_addr . ">\r\n"; + } + + $headers .= 'Content-Type: text/plain; charset="' . get_option( 'blog_charset' ) . '"'; + } + + /** + * Filters the subject of the email sent after a contact form submission. + * + * @module contact-form + * + * @since 3.0.0 + * + * @param string $content_fields['_feedback_subject'] Feedback's subject line. + * @param array $content_fields['_feedback_all_fields'] Feedback's data from old fields. + */ + $subject = apply_filters( 'contact_form_subject', $content_fields['_feedback_subject'], $content_fields['_feedback_all_fields'] ); + + Grunion_Contact_Form::wp_mail( $to, $subject, $message, $headers ); + } + } elseif ( $_POST['make_it'] == 'publish' ) { + if ( ! current_user_can( $post_type_object->cap->delete_post, $post_id ) ) { + wp_die( __( 'You are not allowed to move this item out of the Trash.', 'jetpack' ) ); + } + + if ( ! wp_untrash_post( $post_id ) ) { + wp_die( __( 'Error in restoring from Trash.', 'jetpack' ) ); + } + } elseif ( $_POST['make_it'] == 'trash' ) { + if ( ! current_user_can( $post_type_object->cap->delete_post, $post_id ) ) { + wp_die( __( 'You are not allowed to move this item to the Trash.', 'jetpack' ) ); + } + + if ( ! wp_trash_post( $post_id ) ) { + wp_die( __( 'Error in moving to Trash.', 'jetpack' ) ); + } + } + + $sql = " + SELECT post_status, + COUNT( * ) AS post_count + FROM `{$wpdb->posts}` + WHERE post_type = 'feedback' + GROUP BY post_status + "; + $status_count = (array) $wpdb->get_results( $sql, ARRAY_A ); + + $status = array(); + $status_html = ''; + foreach ( $status_count as $i => $row ) { + $status[ $row['post_status'] ] = $row['post_count']; + } + + if ( isset( $status['publish'] ) ) { + $status_html .= '<li><a href="edit.php?post_type=feedback"'; + if ( $current_menu == 'messages' ) { + $status_html .= ' class="current"'; + } + + $status_html .= '>' . __( 'Messages', 'jetpack' ) . ' <span class="count">'; + $status_html .= '(' . number_format( $status['publish'] ) . ')'; + $status_html .= '</span></a> |</li>'; + } + + if ( isset( $status['trash'] ) ) { + $status_html .= '<li><a href="edit.php?post_status=trash&post_type=feedback"'; + if ( $current_menu == 'trash' ) { + $status_html .= ' class="current"'; + } + + $status_html .= '>' . __( 'Trash', 'jetpack' ) . ' <span class="count">'; + $status_html .= '(' . number_format( $status['trash'] ) . ')'; + $status_html .= '</span></a>'; + if ( isset( $status['spam'] ) ) { + $status_html .= ' |'; + } + $status_html .= '</li>'; + } + + if ( isset( $status['spam'] ) ) { + $status_html .= '<li><a href="edit.php?post_status=spam&post_type=feedback"'; + if ( $current_menu == 'spam' ) { + $status_html .= ' class="current"'; + } + + $status_html .= '>' . __( 'Spam', 'jetpack' ) . ' <span class="count">'; + $status_html .= '(' . number_format( $status['spam'] ) . ')'; + $status_html .= '</span></a></li>'; + } + + echo $status_html; + exit; +} + +/** + * Add the scripts that will add the "Check for Spam" button to the Feedbacks dashboard page. + */ +function grunion_enable_spam_recheck() { + if ( ! defined( 'AKISMET_VERSION' ) ) { + return; + } + + $screen = get_current_screen(); + + // Only add to feedback, only to non-spam view + if ( 'edit-feedback' != $screen->id || ( ! empty( $_GET['post_status'] ) && 'spam' == $_GET['post_status'] ) ) { + return; + } + + // Add the scripts that handle the spam check event. + wp_register_script( + 'grunion-admin', + Jetpack::get_file_url_for_environment( + '_inc/build/contact-form/js/grunion-admin.min.js', + 'modules/contact-form/js/grunion-admin.js' + ), + array( 'jquery' ) + ); + wp_enqueue_script( 'grunion-admin' ); + + wp_enqueue_style( 'grunion.css' ); + + // Add the actual "Check for Spam" button. + add_action( 'admin_head', 'grunion_check_for_spam_button' ); +} + +add_action( 'admin_enqueue_scripts', 'grunion_enable_spam_recheck' ); + +/** + * Add the "Check for Spam" button to the Feedbacks dashboard page. + */ +function grunion_check_for_spam_button() { + // Get HTML for the button + $button_html = get_submit_button( + __( 'Check for Spam', 'jetpack' ), + 'secondary', + 'jetpack-check-feedback-spam', + false, + array( 'class' => 'jetpack-check-feedback-spam' ) + ); + $button_html .= '<span class="jetpack-check-feedback-spam-spinner"></span>'; + + // Add the button next to the filter button via js + ?> + <script type="text/javascript"> + jQuery( function( $ ) { + $( '#posts-filter #post-query-submit' ).after( '<?php echo $button_html; ?>' ); + } ); + </script> + <?php +} + +/** + * Recheck all approved feedbacks for spam. + */ +function grunion_recheck_queue() { + global $wpdb; + + $query = 'post_type=feedback&post_status=publish'; + + if ( isset( $_POST['limit'], $_POST['offset'] ) ) { + $query .= '&posts_per_page=' . intval( $_POST['limit'] ) . '&offset=' . intval( $_POST['offset'] ); + } + + $approved_feedbacks = get_posts( $query ); + + foreach ( $approved_feedbacks as $feedback ) { + $meta = get_post_meta( $feedback->ID, '_feedback_akismet_values', true ); + + /** + * Filter whether the submitted feedback is considered as spam. + * + * @module contact-form + * + * @since 3.4.0 + * + * @param bool false Is the submitted feedback spam? Default to false. + * @param array $meta Feedack values returned by the Akismet plugin. + */ + $is_spam = apply_filters( 'jetpack_contact_form_is_spam', false, $meta ); + + if ( $is_spam ) { + wp_update_post( + array( + 'ID' => $feedback->ID, + 'post_status' => 'spam', + ) + ); + /** This action is already documented in modules/contact-form/admin.php */ + do_action( 'contact_form_akismet', 'spam', $meta ); + } + } + + wp_send_json( + array( + 'processed' => count( $approved_feedbacks ), + ) + ); +} + +add_action( 'wp_ajax_grunion_recheck_queue', 'grunion_recheck_queue' ); diff --git a/plugins/jetpack/modules/contact-form/class-grunion-contact-form-endpoint.php b/plugins/jetpack/modules/contact-form/class-grunion-contact-form-endpoint.php new file mode 100644 index 00000000..ba2785a6 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/class-grunion-contact-form-endpoint.php @@ -0,0 +1,52 @@ +<?php + +/* + * Plugin Name: Feedback CPT Permissions over-ride + */ + +if ( class_exists( 'WP_REST_Posts_Controller' ) ) { + + /** + * Class Grunion_Contact_Form_Endpoint + * Used as 'rest_controller_class' parameter when 'feedback' post type is registered in modules/contact-form/grunion-contact-form.php. + */ + class Grunion_Contact_Form_Endpoint extends WP_REST_Posts_Controller { + /** + * Check whether a given request has proper authorization to view feedback items. + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_Error|boolean + */ + public function get_items_permissions_check( $request ) { + if ( ! is_user_member_of_blog( get_current_user_id(), get_current_blog_id() ) ) { + return new WP_Error( + 'rest_cannot_view', + esc_html__( 'Sorry, you cannot view this resource.', 'jetpack' ), + array( 'status' => 401 ) + ); + } + + return true; + } + + /** + * Check whether a given request has proper authorization to view feedback item. + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_Error|boolean + */ + public function get_item_permissions_check( $request ) { + if ( ! is_user_member_of_blog( get_current_user_id(), get_current_blog_id() ) ) { + return new WP_Error( + 'rest_cannot_view', + esc_html__( 'Sorry, you cannot view this resource.', 'jetpack' ), + array( 'status' => 401 ) + ); + } + + return true; + } + + } + +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style-rtl.css b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style-rtl.css new file mode 100644 index 00000000..1cfbf3ce --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style-rtl.css @@ -0,0 +1,825 @@ +/* Do not modify this file directly. It is concatenated from individual module CSS files. */ +/* ========================================================================== +** Normalize +** ======================================================================== */ + +html { + direction: rtl; +} + +body { + font-family: -apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif; + font-size: 16px; + line-height: 1.4em; + margin: 0; +} + +/* Links */ +a, +a:visited { + color: #0087be; + text-decoration: none; +} + +a:hover, +a:focus, +a:active { + color: $link-highlight; +} + +/* ========================================================================== +** Card +** ======================================================================= */ + +.card, +body { + display: block; + position: relative; + margin: 0 auto 10px auto; + padding: 16px; + box-sizing: border-box; + background: white; + box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3; +} + +body { + margin: 0; + background: #f5f5f5; +} + +.card:after { + content: "."; + display: block; + height: 0; + clear: both; + visibility: hidden; +} + +.card:hover, +.card:focus { + box-shadow: 0 0 0 1px #999, 0 1px 2px #e9eff3; +} + +.card .delete-field { + display: block; + float: left; +} + +@media ( min-width: 481px ) { + .card { + margin-bottom: 16px; + padding: 24px; + } + body { + padding: 24px; + } +} + +.card.is-compact { + margin-bottom: 1px; +} + +@media ( min-width: 481px ) { + .card.is-compact { + margin-bottom: 1px; + padding: 16px 24px; + } +} + +.card > div { + margin-top: 24px; +} + +.card > div:first-child { + margin-top: 0; +} + + +/* ========================================================================== +** Labels +** ======================================================================= */ + +label { + display: block; + font-size: 14px; + font-weight: 600; + margin-bottom: 5px; + margin-top: 8px; +} + +label:first-of-type { + margin-top: 4px; +} + + +/* ========================================================================== +** Text Inputs +** ======================================================================= */ + +input[type="text"], +input[type="tel"], +input[type="email"], +input[type="url"] { + border-radius: 0; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +input[type="text"]:-ms-input-placeholder, +input[type="tel"]:-ms-input-placeholder, +input[type="email"]:-ms-input-placeholder, +input[type="url"]:-ms-input-placeholder { + color: #87a6bc; +} + +input[type="text"]::-ms-input-placeholder, +input[type="tel"]::-ms-input-placeholder, +input[type="email"]::-ms-input-placeholder, +input[type="url"]::-ms-input-placeholder { + color: #87a6bc; +} + +input[type="text"]::placeholder, +input[type="tel"]::placeholder, +input[type="email"]::placeholder, +input[type="url"]::placeholder { + color: #87a6bc; +} + +input[type="text"]:hover, +input[type="tel"]:hover, +input[type="email"]:hover, +input[type="url"]:hover { + border-color: #a8bece; +} + +input[type="text"]:focus, +input[type="tel"]:focus, +input[type="email"]:focus, +input[type="url"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="text"]:focus::-ms-clear, +input[type="tel"]:focus::-ms-clear, +input[type="email"]:focus::-ms-clear, +input[type="url"]:focus::-ms-clear { + display: none; +} + +input[type="text"]:disabled, +input[type="tel"]:disabled, +input[type="email"]:disabled, +input[type="url"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +input[type="text"]:disabled:hover, +input[type="tel"]:disabled:hover, +input[type="email"]:disabled:hover, +input[type="url"]:disabled:hover { + cursor: default; +} + +input[type="text"]:disabled:-ms-input-placeholder, +input[type="tel"]:disabled:-ms-input-placeholder, +input[type="email"]:disabled:-ms-input-placeholder, +input[type="url"]:disabled:-ms-input-placeholder { + color: #a8bece; +} + +input[type="text"]:disabled::-ms-input-placeholder, +input[type="tel"]:disabled::-ms-input-placeholder, +input[type="email"]:disabled::-ms-input-placeholder, +input[type="url"]:disabled::-ms-input-placeholder { + color: #a8bece; +} + +input[type="text"]:disabled::placeholder, +input[type="tel"]:disabled::placeholder, +input[type="email"]:disabled::placeholder, +input[type="url"]:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Textareas +** ======================================================================= */ + +textarea { + border-radius: 0; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + height: 92px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +textarea:-ms-input-placeholder { + color: #87a6bc; +} + +textarea::-ms-input-placeholder { + color: #87a6bc; +} + +textarea::placeholder { + color: #87a6bc; +} + +textarea:hover { + border-color: #a8bece; +} + +textarea:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +textarea:focus::-ms-clear { + display: none; +} + +textarea:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +textarea:disabled:hover { + cursor: default; +} + +textarea:disabled:-ms-input-placeholder { + color: #a8bece; +} + +textarea:disabled::-ms-input-placeholder { + color: #a8bece; +} + +textarea:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Checkboxes +** ======================================================================= */ + +.checkbox, +input[type="checkbox"] { + -webkit-appearance: none; + display: inline-block; + box-sizing: border-box; + margin: 2px 0 0; + padding: 7px 14px; + width: 16px; + height: 16px; + float: right; + outline: 0; + padding: 0; + box-shadow: none; + background-color: #fff; + border: 1px solid #c8d7e1; + color: #2e4453; + font-size: 16px; + line-height: 0; + text-align: center; + vertical-align: middle; + -moz-appearance: none; + appearance: none; + transition: all .15s ease-in-out; + clear: none; + cursor: pointer; +} + +.checkbox:checked:before, +input[type="checkbox"]:checked:before { + content: '\f147'; + font-family: Dashicons; + margin: -3px -4px 0 0; + float: right; + display: inline-block; + vertical-align: middle; + width: 16px; + font-size: 20px; + line-height: 1; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + speak: none; + color: #00aadc; +} + +.checkbox:disabled:checked:before, +input[type="checkbox"]:disabled:checked:before { + color: #a8bece; +} + +.checkbox:hover, +input[type="checkbox"]:hover { + border-color: #a8bece; +} + +.checkbox:focus, +input[type="checkbox"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +.checkbox:disabled, +input[type="checkbox"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; +} + +.checkbox:disabled:hover, +input[type="checkbox"]:disabled:hover { + cursor: default; +} + +.checkbox + span, +input[type="checkbox"] + span { + display: block; + font-weight: normal; + margin-right: 24px; +} + + +/* ========================================================================== +** Radio buttons +** ======================================================================== */ + +.radio-button, +input[type=radio] { + color: #2e4453; + font-size: 16px; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-sizing: border-box; + -webkit-appearance: none; + clear: none; + cursor: pointer; + display: inline-block; + line-height: 0; + height: 16px; + margin: 2px 0 0 4px; + float: right; + outline: 0; + padding: 0; + text-align: center; + vertical-align: middle; + width: 16px; + min-width: 16px; + -moz-appearance: none; + appearance: none; + border-radius: 50%; + line-height: 10px; +} + +.radio-button:hover, +input[type="radio"]:hover { + border-color: #a8bece; +} + +.radio-button:focus, +input[type="radio"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +.radio-button:focus::-ms-clear, +input[type="radio"]:focus::-ms-clear { + display: none; +} + +.radio-button:checked:before, +input[type="radio"]:checked:before { + float: right; + display: inline-block; + content: '\2022'; + margin: 3px; + width: 8px; + height: 8px; + text-indent: -9999px; + background: #00aadc; + vertical-align: middle; + border-radius: 50%; + animation: grow .2s ease-in-out; +} + +.radio-button:disabled, +input[type="radio"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; + -webkit-text-fill-color: #a8bece; +} + +.radio-button:disabled:hover, +input[type="radio"]:disabled:hover { + cursor: default; +} + +.radio-button:disabled:-ms-input-placeholder, +input[type="radio"]:disabled:-ms-input-placeholder { + color: #a8bece; +} + +.radio-button:disabled::-ms-input-placeholder, +input[type="radio"]:disabled::-ms-input-placeholder { + color: #a8bece; +} + +.radio-button:disabled::placeholder, +input[type="radio"]:disabled::placeholder { + color: #a8bece; +} + +.radio-button:disabled:checked::before, +input[type="radio"]:disabled:checked:before { + background: #e9eff3; +} + +.radio-button + span, +input[type="radio"] + span { + display: block; + font-weight: normal; + margin-right: 24px; +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + + +/* ========================================================================== +** Selects +** ======================================================================== */ + +select { + background: #fff url() no-repeat left 10px center; + border-color: #c8d7e1; + border-style: solid; + border-radius: 4px; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 0; + outline: 0; + overflow: hidden; + font-size: 14px; + line-height: 21px; + font-weight: 600; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + white-space: nowrap; + box-sizing: border-box; + /* Aligns the text to the 8px baseline grid and adds padding on right to allow for the arrow. */ + padding: 7px 14px 9px 32px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +select:hover { + background-image: url(); +} + +select:focus { + background-image: url(); + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; + outline: 0; + -moz-outline:none; + -moz-user-focus:ignore; +} + +select:disabled, +select:hover:disabled { + background: url() no-repeat left 10px center;; +} + +select.is-compact { + min-width: 0; + padding: 0 6px 2px 20px; + margin: 0 4px; + background-position: left 5px center; + background-size: 12px 12px; +} + +/* Make it display:block when it follows a label */ +label select, +label + select { + display: block; + min-width: 200px; +} + +label select.is-compact, +label + select.is-compact { + display: inline-block; + min-width: 0; +} + +/* IE: Remove the default arrow */ +select::-ms-expand { + display: none; +} + +/* IE: Remove default background and color styles on focus */ +select::-ms-value { + background: none; + color: #2e4453; +} + +/* Firefox: Remove the focus outline, see http://stackoverflow.com/questions/3773430/remove-outline-from-select-box-in-ff/18853002#18853002 */ +select:-moz-focusring { + color: transparent; + text-shadow: 0 0 0 #2e4453; +} + + +/* ========================================================================== +** Buttons +** ======================================================================== */ + +input[type="submit"] { + padding: 0; + font-size: 14px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + vertical-align: baseline; + background: white; + border-color: #c8d7e1; + border-style: solid; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 24px 0 0; + outline: 0; + overflow: hidden; + font-weight: 500; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + box-sizing: border-box; + font-size: 14px; + line-height: 21px; + border-radius: 4px; + padding: 7px 14px 9px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +input[type="submit"]:hover { + border-color: #a8bece; + color: #2e4453; +} + +input[type="submit"]:active { + border-width: 2px 1px 1px; +} + +input[type="submit"]:visited { + color: #2e4453; +} + +input[type="submit"][disabled], +input[type="submit"]:disabled { + color: #e9eff3; + background: white; + border-color: #e9eff3; + cursor: default; +} + +input[type="submit"][disabled]:active, +input[type="submit"]:disabled:active { + border-width: 1px 1px 2px; +} + +input[type="submit"]:focus { + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="submit"].hidden { + display: none; +} + +input[type="submit"] .gridicon { + position: relative; + top: 4px; + margin-top: -2px; + width: 18px; + height: 18px; +} + +input[type="submit"].button-primary { + background: #00aadc; + border-color: #008ab3; + color: white; +} + +input[type="submit"].button-primary:hover, +input[type="submit"].button-primary:focus { + border-color: #005082; + color: white; +} + +input[type="submit"].button-primary[disabled], +input[type="submit"].button-primary:disabled { + background: #bceefd; + border-color: #8cc9e2; + color: white; +} + +input[type="submit"].button-primary { + color: white; +} + + +/* ========================================================================== +** Inline editor styles +** ======================================================================== */ + + +.ui-sortable-handle { + cursor: move; +} + +.grunion-section-header { + font-size: 21px; + margin-top: 32px; + font-weight: 600; +} + +.grunion-form-settings:hover { + box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3; +} + +.grunion-section-header:first-child { + margin-top: 0; +} + +.grunion-type-options { + display: flex; + flex-wrap: wrap; +} + +.grunion-type { + flex-grow: 0; + flex-shrink: 0; +} + +.grunion-type select { + -webkit-appearance: none; + width: 100%; +} + +.grunion-required { + padding: 27px 16px 0 0; + flex-grow: 0; + flex-shrink: 0; +} + +.grunion-options { + padding-top: 16px; +} + +.grunion-options ol { + list-style: none; + padding: 0; + margin: 8px 0 0; +} + +.grunion-options li { + display: flex; + margin-bottom: 16px; +} + +.grunion-field-edit .grunion-options { + display: none; +} + +.delete-option, +.delete-field { + color: #0087be; + text-decoration: none; + width: 40px; + line-height: 40px; + font-size: 21px; + text-align: center; + font-weight: 600; +} + +.delete-field { + position: absolute; + top: 0; + left: 0; +} + +.grunion-controls { + display: flex; + flex-wrap: wrap; +} + +.grunion-update-controls { + text-align: left; + flex-grow: 1; +} + +#add-field { + flex-grow: 0; +} + +.delete-option:before, +.delete-field:before { + font-family: Dashicons; +/* content: "\f158"; /* This is the bolder X */ + content: "\f335"; /* This is the thinner X */ + display: inline-block; + speak: none; +} + +.grunion-field-edit.grunion-field-checkbox-multiple .grunion-options, +.grunion-field-edit.grunion-field-radio .grunion-options, +.grunion-field-edit.grunion-field-select .grunion-options { + display: block; +} + +.screen-reader-text { + position: absolute; + margin: -1px; + padding: 0; + height: 1px; + width: 1px; + overflow: hidden; + clip: rect(0 0 0 0); + border: 0; + word-wrap: normal !important; /* many screen reader and browser combinations announce broken words as they would appear visually */ +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style-rtl.min.css b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style-rtl.min.css new file mode 100644 index 00000000..9b7a27c2 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style-rtl.min.css @@ -0,0 +1 @@ +html{direction:rtl}body{font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif;font-size:16px;line-height:1.4em;margin:0}a,a:visited{color:#0087be;text-decoration:none}a:active,a:focus,a:hover{color:$link-highlight}.card,body{display:block;position:relative;margin:0 auto 10px auto;padding:16px;box-sizing:border-box;background:#fff;box-shadow:0 0 0 1px rgba(200,215,225,.5),0 1px 2px #e9eff3}body{margin:0;background:#f5f5f5}.card:after{content:".";display:block;height:0;clear:both;visibility:hidden}.card:focus,.card:hover{box-shadow:0 0 0 1px #999,0 1px 2px #e9eff3}.card .delete-field{display:block;float:left}@media (min-width:481px){.card{margin-bottom:16px;padding:24px}body{padding:24px}}.card.is-compact{margin-bottom:1px}@media (min-width:481px){.card.is-compact{margin-bottom:1px;padding:16px 24px}}.card>div{margin-top:24px}.card>div:first-child{margin-top:0}label{display:block;font-size:14px;font-weight:600;margin-bottom:5px;margin-top:8px}label:first-of-type{margin-top:4px}input[type=email],input[type=tel],input[type=text],input[type=url]{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}input[type=email]:-ms-input-placeholder,input[type=tel]:-ms-input-placeholder,input[type=text]:-ms-input-placeholder,input[type=url]:-ms-input-placeholder{color:#87a6bc}input[type=email]::-ms-input-placeholder,input[type=tel]::-ms-input-placeholder,input[type=text]::-ms-input-placeholder,input[type=url]::-ms-input-placeholder{color:#87a6bc}input[type=email]::placeholder,input[type=tel]::placeholder,input[type=text]::placeholder,input[type=url]::placeholder{color:#87a6bc}input[type=email]:hover,input[type=tel]:hover,input[type=text]:hover,input[type=url]:hover{border-color:#a8bece}input[type=email]:focus,input[type=tel]:focus,input[type=text]:focus,input[type=url]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=email]:focus::-ms-clear,input[type=tel]:focus::-ms-clear,input[type=text]:focus::-ms-clear,input[type=url]:focus::-ms-clear{display:none}input[type=email]:disabled,input[type=tel]:disabled,input[type=text]:disabled,input[type=url]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}input[type=email]:disabled:hover,input[type=tel]:disabled:hover,input[type=text]:disabled:hover,input[type=url]:disabled:hover{cursor:default}input[type=email]:disabled:-ms-input-placeholder,input[type=tel]:disabled:-ms-input-placeholder,input[type=text]:disabled:-ms-input-placeholder,input[type=url]:disabled:-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::-ms-input-placeholder,input[type=tel]:disabled::-ms-input-placeholder,input[type=text]:disabled::-ms-input-placeholder,input[type=url]:disabled::-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::placeholder,input[type=tel]:disabled::placeholder,input[type=text]:disabled::placeholder,input[type=url]:disabled::placeholder{color:#a8bece}textarea{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;height:92px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}textarea:-ms-input-placeholder{color:#87a6bc}textarea::-ms-input-placeholder{color:#87a6bc}textarea::placeholder{color:#87a6bc}textarea:hover{border-color:#a8bece}textarea:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}textarea:focus::-ms-clear{display:none}textarea:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}textarea:disabled:hover{cursor:default}textarea:disabled:-ms-input-placeholder{color:#a8bece}textarea:disabled::-ms-input-placeholder{color:#a8bece}textarea:disabled::placeholder{color:#a8bece}.checkbox,input[type=checkbox]{-webkit-appearance:none;display:inline-block;box-sizing:border-box;margin:2px 0 0;padding:7px 14px;width:16px;height:16px;float:right;outline:0;padding:0;box-shadow:none;background-color:#fff;border:1px solid #c8d7e1;color:#2e4453;font-size:16px;line-height:0;text-align:center;vertical-align:middle;-moz-appearance:none;appearance:none;transition:all .15s ease-in-out;clear:none;cursor:pointer}.checkbox:checked:before,input[type=checkbox]:checked:before{content:'\f147';font-family:Dashicons;margin:-3px -4px 0 0;float:right;display:inline-block;vertical-align:middle;width:16px;font-size:20px;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;speak:none;color:#00aadc}.checkbox:disabled:checked:before,input[type=checkbox]:disabled:checked:before{color:#a8bece}.checkbox:hover,input[type=checkbox]:hover{border-color:#a8bece}.checkbox:focus,input[type=checkbox]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}.checkbox:disabled,input[type=checkbox]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1}.checkbox:disabled:hover,input[type=checkbox]:disabled:hover{cursor:default}.checkbox+span,input[type=checkbox]+span{display:block;font-weight:400;margin-right:24px}.radio-button,input[type=radio]{color:#2e4453;font-size:16px;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-sizing:border-box;-webkit-appearance:none;clear:none;cursor:pointer;display:inline-block;line-height:0;height:16px;margin:2px 0 0 4px;float:right;outline:0;padding:0;text-align:center;vertical-align:middle;width:16px;min-width:16px;-moz-appearance:none;appearance:none;border-radius:50%;line-height:10px}.radio-button:hover,input[type=radio]:hover{border-color:#a8bece}.radio-button:focus,input[type=radio]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}.radio-button:focus::-ms-clear,input[type=radio]:focus::-ms-clear{display:none}.radio-button:checked:before,input[type=radio]:checked:before{float:right;display:inline-block;content:'\2022';margin:3px;width:8px;height:8px;text-indent:-9999px;background:#00aadc;vertical-align:middle;border-radius:50%;animation:grow .2s ease-in-out}.radio-button:disabled,input[type=radio]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1;-webkit-text-fill-color:#a8bece}.radio-button:disabled:hover,input[type=radio]:disabled:hover{cursor:default}.radio-button:disabled:-ms-input-placeholder,input[type=radio]:disabled:-ms-input-placeholder{color:#a8bece}.radio-button:disabled::-ms-input-placeholder,input[type=radio]:disabled::-ms-input-placeholder{color:#a8bece}.radio-button:disabled::placeholder,input[type=radio]:disabled::placeholder{color:#a8bece}.radio-button:disabled:checked::before,input[type=radio]:disabled:checked:before{background:#e9eff3}.radio-button+span,input[type=radio]+span{display:block;font-weight:400;margin-right:24px}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}select{background:#fff url() no-repeat left 10px center;border-color:#c8d7e1;border-style:solid;border-radius:4px;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:0;outline:0;overflow:hidden;font-size:14px;line-height:21px;font-weight:600;text-overflow:ellipsis;text-decoration:none;vertical-align:top;white-space:nowrap;box-sizing:border-box;padding:7px 14px 9px 32px;-webkit-appearance:none;-moz-appearance:none;appearance:none}select:hover{background-image:url()}select:focus{background-image:url();border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa;outline:0;-moz-outline:none;-moz-user-focus:ignore}select:disabled,select:hover:disabled{background:url() no-repeat left 10px center}select.is-compact{min-width:0;padding:0 6px 2px 20px;margin:0 4px;background-position:left 5px center;background-size:12px 12px}label select,label+select{display:block;min-width:200px}label select.is-compact,label+select.is-compact{display:inline-block;min-width:0}select::-ms-expand{display:none}select::-ms-value{background:0 0;color:#2e4453}select:-moz-focusring{color:transparent;text-shadow:0 0 0 #2e4453}input[type=submit]{padding:0;font-size:14px;-webkit-appearance:none;-moz-appearance:none;appearance:none;vertical-align:baseline;background:#fff;border-color:#c8d7e1;border-style:solid;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:24px 0 0;outline:0;overflow:hidden;font-weight:500;text-overflow:ellipsis;text-decoration:none;vertical-align:top;box-sizing:border-box;font-size:14px;line-height:21px;border-radius:4px;padding:7px 14px 9px;-webkit-appearance:none;-moz-appearance:none;appearance:none}input[type=submit]:hover{border-color:#a8bece;color:#2e4453}input[type=submit]:active{border-width:2px 1px 1px}input[type=submit]:visited{color:#2e4453}input[type=submit]:disabled,input[type=submit][disabled]{color:#e9eff3;background:#fff;border-color:#e9eff3;cursor:default}input[type=submit]:disabled:active,input[type=submit][disabled]:active{border-width:1px 1px 2px}input[type=submit]:focus{border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa}input[type=submit].hidden{display:none}input[type=submit] .gridicon{position:relative;top:4px;margin-top:-2px;width:18px;height:18px}input[type=submit].button-primary{background:#00aadc;border-color:#008ab3;color:#fff}input[type=submit].button-primary:focus,input[type=submit].button-primary:hover{border-color:#005082;color:#fff}input[type=submit].button-primary:disabled,input[type=submit].button-primary[disabled]{background:#bceefd;border-color:#8cc9e2;color:#fff}input[type=submit].button-primary{color:#fff}.ui-sortable-handle{cursor:move}.grunion-section-header{font-size:21px;margin-top:32px;font-weight:600}.grunion-form-settings:hover{box-shadow:0 0 0 1px rgba(200,215,225,.5),0 1px 2px #e9eff3}.grunion-section-header:first-child{margin-top:0}.grunion-type-options{display:flex;flex-wrap:wrap}.grunion-type{flex-grow:0;flex-shrink:0}.grunion-type select{-webkit-appearance:none;width:100%}.grunion-required{padding:27px 16px 0 0;flex-grow:0;flex-shrink:0}.grunion-options{padding-top:16px}.grunion-options ol{list-style:none;padding:0;margin:8px 0 0}.grunion-options li{display:flex;margin-bottom:16px}.grunion-field-edit .grunion-options{display:none}.delete-field,.delete-option{color:#0087be;text-decoration:none;width:40px;line-height:40px;font-size:21px;text-align:center;font-weight:600}.delete-field{position:absolute;top:0;left:0}.grunion-controls{display:flex;flex-wrap:wrap}.grunion-update-controls{text-align:left;flex-grow:1}#add-field{flex-grow:0}.delete-field:before,.delete-option:before{font-family:Dashicons;content:"\f335";display:inline-block;speak:none}.grunion-field-edit.grunion-field-checkbox-multiple .grunion-options,.grunion-field-edit.grunion-field-radio .grunion-options,.grunion-field-edit.grunion-field-select .grunion-options{display:block}.screen-reader-text{position:absolute;margin:-1px;padding:0;height:1px;width:1px;overflow:hidden;clip:rect(0 0 0 0);border:0;word-wrap:normal!important}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style.css b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style.css new file mode 100644 index 00000000..13db02fd --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style.css @@ -0,0 +1,764 @@ +/* ========================================================================== +** Normalize +** ======================================================================== */ + +html { + direction: ltr; +} + +body { + font-family: -apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif; + font-size: 16px; + line-height: 1.4em; + margin: 0; +} + +/* Links */ +a, +a:visited { + color: #0087be; + text-decoration: none; +} + +a:hover, +a:focus, +a:active { + color: $link-highlight; +} + +/* ========================================================================== +** Card +** ======================================================================= */ + +.card, +body { + display: block; + position: relative; + margin: 0 auto 10px auto; + padding: 16px; + box-sizing: border-box; + background: white; + box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3; +} + +body { + margin: 0; + background: #f5f5f5; +} + +.card:after { + content: "."; + display: block; + height: 0; + clear: both; + visibility: hidden; +} + +.card:hover, +.card:focus { + box-shadow: 0 0 0 1px #999, 0 1px 2px #e9eff3; +} + +.card .delete-field { + display: block; + float: right; +} + +@media ( min-width: 481px ) { + .card { + margin-bottom: 16px; + padding: 24px; + } + body { + padding: 24px; + } +} + +.card.is-compact { + margin-bottom: 1px; +} + +@media ( min-width: 481px ) { + .card.is-compact { + margin-bottom: 1px; + padding: 16px 24px; + } +} + +.card > div { + margin-top: 24px; +} + +.card > div:first-child { + margin-top: 0; +} + + +/* ========================================================================== +** Labels +** ======================================================================= */ + +label { + display: block; + font-size: 14px; + font-weight: 600; + margin-bottom: 5px; + margin-top: 8px; +} + +label:first-of-type { + margin-top: 4px; +} + + +/* ========================================================================== +** Text Inputs +** ======================================================================= */ + +input[type="text"], +input[type="tel"], +input[type="email"], +input[type="url"] { + border-radius: 0; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +input[type="text"]::placeholder, +input[type="tel"]::placeholder, +input[type="email"]::placeholder, +input[type="url"]::placeholder { + color: #87a6bc; +} + +input[type="text"]:hover, +input[type="tel"]:hover, +input[type="email"]:hover, +input[type="url"]:hover { + border-color: #a8bece; +} + +input[type="text"]:focus, +input[type="tel"]:focus, +input[type="email"]:focus, +input[type="url"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="text"]:focus::-ms-clear, +input[type="tel"]:focus::-ms-clear, +input[type="email"]:focus::-ms-clear, +input[type="url"]:focus::-ms-clear { + display: none; +} + +input[type="text"]:disabled, +input[type="tel"]:disabled, +input[type="email"]:disabled, +input[type="url"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +input[type="text"]:disabled:hover, +input[type="tel"]:disabled:hover, +input[type="email"]:disabled:hover, +input[type="url"]:disabled:hover { + cursor: default; +} + +input[type="text"]:disabled::placeholder, +input[type="tel"]:disabled::placeholder, +input[type="email"]:disabled::placeholder, +input[type="url"]:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Textareas +** ======================================================================= */ + +textarea { + border-radius: 0; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + height: 92px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +textarea::placeholder { + color: #87a6bc; +} + +textarea:hover { + border-color: #a8bece; +} + +textarea:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +textarea:focus::-ms-clear { + display: none; +} + +textarea:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +textarea:disabled:hover { + cursor: default; +} + +textarea:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Checkboxes +** ======================================================================= */ + +.checkbox, +input[type="checkbox"] { + -webkit-appearance: none; + display: inline-block; + box-sizing: border-box; + margin: 2px 0 0; + padding: 7px 14px; + width: 16px; + height: 16px; + float: left; + outline: 0; + padding: 0; + box-shadow: none; + background-color: #fff; + border: 1px solid #c8d7e1; + color: #2e4453; + font-size: 16px; + line-height: 0; + text-align: center; + vertical-align: middle; + appearance: none; + transition: all .15s ease-in-out; + clear: none; + cursor: pointer; +} + +.checkbox:checked:before, +input[type="checkbox"]:checked:before { + content: '\f147'; + font-family: Dashicons; + margin: -3px 0 0 -4px; + float: left; + display: inline-block; + vertical-align: middle; + width: 16px; + font-size: 20px; + line-height: 1; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + speak: none; + color: #00aadc; +} + +.checkbox:disabled:checked:before, +input[type="checkbox"]:disabled:checked:before { + color: #a8bece; +} + +.checkbox:hover, +input[type="checkbox"]:hover { + border-color: #a8bece; +} + +.checkbox:focus, +input[type="checkbox"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +.checkbox:disabled, +input[type="checkbox"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; +} + +.checkbox:disabled:hover, +input[type="checkbox"]:disabled:hover { + cursor: default; +} + +.checkbox + span, +input[type="checkbox"] + span { + display: block; + font-weight: normal; + margin-left: 24px; +} + + +/* ========================================================================== +** Radio buttons +** ======================================================================== */ + +.radio-button, +input[type=radio] { + color: #2e4453; + font-size: 16px; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-sizing: border-box; + -webkit-appearance: none; + clear: none; + cursor: pointer; + display: inline-block; + line-height: 0; + height: 16px; + margin: 2px 4px 0 0; + float: left; + outline: 0; + padding: 0; + text-align: center; + vertical-align: middle; + width: 16px; + min-width: 16px; + appearance: none; + border-radius: 50%; + line-height: 10px; +} + +.radio-button:hover, +input[type="radio"]:hover { + border-color: #a8bece; +} + +.radio-button:focus, +input[type="radio"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +.radio-button:focus::-ms-clear, +input[type="radio"]:focus::-ms-clear { + display: none; +} + +.radio-button:checked:before, +input[type="radio"]:checked:before { + float: left; + display: inline-block; + content: '\2022'; + margin: 3px; + width: 8px; + height: 8px; + text-indent: -9999px; + background: #00aadc; + vertical-align: middle; + border-radius: 50%; + animation: grow .2s ease-in-out; +} + +.radio-button:disabled, +input[type="radio"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; + -webkit-text-fill-color: #a8bece; +} + +.radio-button:disabled:hover, +input[type="radio"]:disabled:hover { + cursor: default; +} + +.radio-button:disabled::placeholder, +input[type="radio"]:disabled::placeholder { + color: #a8bece; +} + +.radio-button:disabled:checked::before, +input[type="radio"]:disabled:checked:before { + background: #e9eff3; +} + +.radio-button + span, +input[type="radio"] + span { + display: block; + font-weight: normal; + margin-left: 24px; +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + + +/* ========================================================================== +** Selects +** ======================================================================== */ + +select { + background: #fff url() no-repeat right 10px center; + border-color: #c8d7e1; + border-style: solid; + border-radius: 4px; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 0; + outline: 0; + overflow: hidden; + font-size: 14px; + line-height: 21px; + font-weight: 600; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + white-space: nowrap; + box-sizing: border-box; + /* Aligns the text to the 8px baseline grid and adds padding on right to allow for the arrow. */ + padding: 7px 32px 9px 14px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +select:hover { + background-image: url(); +} + +select:focus { + background-image: url(); + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; + outline: 0; + -moz-outline:none; + -moz-user-focus:ignore; +} + +select:disabled, +select:hover:disabled { + background: url() no-repeat right 10px center;; +} + +select.is-compact { + min-width: 0; + padding: 0 20px 2px 6px; + margin: 0 4px; + background-position: right 5px center; + background-size: 12px 12px; +} + +/* Make it display:block when it follows a label */ +label select, +label + select { + display: block; + min-width: 200px; +} + +label select.is-compact, +label + select.is-compact { + display: inline-block; + min-width: 0; +} + +/* IE: Remove the default arrow */ +select::-ms-expand { + display: none; +} + +/* IE: Remove default background and color styles on focus */ +select::-ms-value { + background: none; + color: #2e4453; +} + +/* Firefox: Remove the focus outline, see http://stackoverflow.com/questions/3773430/remove-outline-from-select-box-in-ff/18853002#18853002 */ +select:-moz-focusring { + color: transparent; + text-shadow: 0 0 0 #2e4453; +} + + +/* ========================================================================== +** Buttons +** ======================================================================== */ + +input[type="submit"] { + padding: 0; + font-size: 14px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + vertical-align: baseline; + background: white; + border-color: #c8d7e1; + border-style: solid; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 24px 0 0; + outline: 0; + overflow: hidden; + font-weight: 500; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + box-sizing: border-box; + font-size: 14px; + line-height: 21px; + border-radius: 4px; + padding: 7px 14px 9px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +input[type="submit"]:hover { + border-color: #a8bece; + color: #2e4453; +} + +input[type="submit"]:active { + border-width: 2px 1px 1px; +} + +input[type="submit"]:visited { + color: #2e4453; +} + +input[type="submit"][disabled], +input[type="submit"]:disabled { + color: #e9eff3; + background: white; + border-color: #e9eff3; + cursor: default; +} + +input[type="submit"][disabled]:active, +input[type="submit"]:disabled:active { + border-width: 1px 1px 2px; +} + +input[type="submit"]:focus { + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="submit"].hidden { + display: none; +} + +input[type="submit"] .gridicon { + position: relative; + top: 4px; + margin-top: -2px; + width: 18px; + height: 18px; +} + +input[type="submit"].button-primary { + background: #00aadc; + border-color: #008ab3; + color: white; +} + +input[type="submit"].button-primary:hover, +input[type="submit"].button-primary:focus { + border-color: #005082; + color: white; +} + +input[type="submit"].button-primary[disabled], +input[type="submit"].button-primary:disabled { + background: #bceefd; + border-color: #8cc9e2; + color: white; +} + +input[type="submit"].button-primary { + color: white; +} + + +/* ========================================================================== +** Inline editor styles +** ======================================================================== */ + + +.ui-sortable-handle { + cursor: move; +} + +.grunion-section-header { + font-size: 21px; + margin-top: 32px; + font-weight: 600; +} + +.grunion-form-settings:hover { + box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3; +} + +.grunion-section-header:first-child { + margin-top: 0; +} + +.grunion-type-options { + display: flex; + flex-wrap: wrap; +} + +.grunion-type { + flex-grow: 0; + flex-shrink: 0; +} + +.grunion-type select { + -webkit-appearance: none; + width: 100%; +} + +.grunion-required { + padding: 27px 0 0 16px; + flex-grow: 0; + flex-shrink: 0; +} + +.grunion-options { + padding-top: 16px; +} + +.grunion-options ol { + list-style: none; + padding: 0; + margin: 8px 0 0; +} + +.grunion-options li { + display: flex; + margin-bottom: 16px; +} + +.grunion-field-edit .grunion-options { + display: none; +} + +.delete-option, +.delete-field { + color: #0087be; + text-decoration: none; + width: 40px; + line-height: 40px; + font-size: 21px; + text-align: center; + font-weight: 600; +} + +.delete-field { + position: absolute; + top: 0; + right: 0; +} + +.grunion-controls { + display: flex; + flex-wrap: wrap; +} + +.grunion-update-controls { + text-align: right; + flex-grow: 1; +} + +#add-field { + flex-grow: 0; +} + +.delete-option:before, +.delete-field:before { + font-family: Dashicons; +/* content: "\f158"; /* This is the bolder X */ + content: "\f335"; /* This is the thinner X */ + display: inline-block; + speak: none; +} + +.grunion-field-edit.grunion-field-checkbox-multiple .grunion-options, +.grunion-field-edit.grunion-field-radio .grunion-options, +.grunion-field-edit.grunion-field-select .grunion-options { + display: block; +} + +.screen-reader-text { + position: absolute; + margin: -1px; + padding: 0; + height: 1px; + width: 1px; + overflow: hidden; + clip: rect(0 0 0 0); + border: 0; + word-wrap: normal !important; /* many screen reader and browser combinations announce broken words as they would appear visually */ +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style.min.css b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style.min.css new file mode 100644 index 00000000..7d3a0ca0 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-inline-editing-style.min.css @@ -0,0 +1,2 @@ +/* Do not modify this file directly. It is concatenated from individual module CSS files. */ +html{direction:ltr}body{font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif;font-size:16px;line-height:1.4em;margin:0}a,a:visited{color:#0087be;text-decoration:none}a:active,a:focus,a:hover{color:$link-highlight}.card,body{display:block;position:relative;margin:0 auto 10px auto;padding:16px;box-sizing:border-box;background:#fff;box-shadow:0 0 0 1px rgba(200,215,225,.5),0 1px 2px #e9eff3}body{margin:0;background:#f5f5f5}.card:after{content:".";display:block;height:0;clear:both;visibility:hidden}.card:focus,.card:hover{box-shadow:0 0 0 1px #999,0 1px 2px #e9eff3}.card .delete-field{display:block;float:right}@media (min-width:481px){.card{margin-bottom:16px;padding:24px}body{padding:24px}}.card.is-compact{margin-bottom:1px}@media (min-width:481px){.card.is-compact{margin-bottom:1px;padding:16px 24px}}.card>div{margin-top:24px}.card>div:first-child{margin-top:0}label{display:block;font-size:14px;font-weight:600;margin-bottom:5px;margin-top:8px}label:first-of-type{margin-top:4px}input[type=email],input[type=tel],input[type=text],input[type=url]{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}input[type=email]:-ms-input-placeholder,input[type=tel]:-ms-input-placeholder,input[type=text]:-ms-input-placeholder,input[type=url]:-ms-input-placeholder{color:#87a6bc}input[type=email]::-ms-input-placeholder,input[type=tel]::-ms-input-placeholder,input[type=text]::-ms-input-placeholder,input[type=url]::-ms-input-placeholder{color:#87a6bc}input[type=email]::placeholder,input[type=tel]::placeholder,input[type=text]::placeholder,input[type=url]::placeholder{color:#87a6bc}input[type=email]:hover,input[type=tel]:hover,input[type=text]:hover,input[type=url]:hover{border-color:#a8bece}input[type=email]:focus,input[type=tel]:focus,input[type=text]:focus,input[type=url]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=email]:focus::-ms-clear,input[type=tel]:focus::-ms-clear,input[type=text]:focus::-ms-clear,input[type=url]:focus::-ms-clear{display:none}input[type=email]:disabled,input[type=tel]:disabled,input[type=text]:disabled,input[type=url]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}input[type=email]:disabled:hover,input[type=tel]:disabled:hover,input[type=text]:disabled:hover,input[type=url]:disabled:hover{cursor:default}input[type=email]:disabled:-ms-input-placeholder,input[type=tel]:disabled:-ms-input-placeholder,input[type=text]:disabled:-ms-input-placeholder,input[type=url]:disabled:-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::-ms-input-placeholder,input[type=tel]:disabled::-ms-input-placeholder,input[type=text]:disabled::-ms-input-placeholder,input[type=url]:disabled::-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::placeholder,input[type=tel]:disabled::placeholder,input[type=text]:disabled::placeholder,input[type=url]:disabled::placeholder{color:#a8bece}textarea{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;height:92px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}textarea:-ms-input-placeholder{color:#87a6bc}textarea::-ms-input-placeholder{color:#87a6bc}textarea::placeholder{color:#87a6bc}textarea:hover{border-color:#a8bece}textarea:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}textarea:focus::-ms-clear{display:none}textarea:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}textarea:disabled:hover{cursor:default}textarea:disabled:-ms-input-placeholder{color:#a8bece}textarea:disabled::-ms-input-placeholder{color:#a8bece}textarea:disabled::placeholder{color:#a8bece}.checkbox,input[type=checkbox]{-webkit-appearance:none;display:inline-block;box-sizing:border-box;margin:2px 0 0;padding:7px 14px;width:16px;height:16px;float:left;outline:0;padding:0;box-shadow:none;background-color:#fff;border:1px solid #c8d7e1;color:#2e4453;font-size:16px;line-height:0;text-align:center;vertical-align:middle;-moz-appearance:none;appearance:none;transition:all .15s ease-in-out;clear:none;cursor:pointer}.checkbox:checked:before,input[type=checkbox]:checked:before{content:'\f147';font-family:Dashicons;margin:-3px 0 0 -4px;float:left;display:inline-block;vertical-align:middle;width:16px;font-size:20px;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;speak:none;color:#00aadc}.checkbox:disabled:checked:before,input[type=checkbox]:disabled:checked:before{color:#a8bece}.checkbox:hover,input[type=checkbox]:hover{border-color:#a8bece}.checkbox:focus,input[type=checkbox]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}.checkbox:disabled,input[type=checkbox]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1}.checkbox:disabled:hover,input[type=checkbox]:disabled:hover{cursor:default}.checkbox+span,input[type=checkbox]+span{display:block;font-weight:400;margin-left:24px}.radio-button,input[type=radio]{color:#2e4453;font-size:16px;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-sizing:border-box;-webkit-appearance:none;clear:none;cursor:pointer;display:inline-block;line-height:0;height:16px;margin:2px 4px 0 0;float:left;outline:0;padding:0;text-align:center;vertical-align:middle;width:16px;min-width:16px;-moz-appearance:none;appearance:none;border-radius:50%;line-height:10px}.radio-button:hover,input[type=radio]:hover{border-color:#a8bece}.radio-button:focus,input[type=radio]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}.radio-button:focus::-ms-clear,input[type=radio]:focus::-ms-clear{display:none}.radio-button:checked:before,input[type=radio]:checked:before{float:left;display:inline-block;content:'\2022';margin:3px;width:8px;height:8px;text-indent:-9999px;background:#00aadc;vertical-align:middle;border-radius:50%;animation:grow .2s ease-in-out}.radio-button:disabled,input[type=radio]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1;-webkit-text-fill-color:#a8bece}.radio-button:disabled:hover,input[type=radio]:disabled:hover{cursor:default}.radio-button:disabled:-ms-input-placeholder,input[type=radio]:disabled:-ms-input-placeholder{color:#a8bece}.radio-button:disabled::-ms-input-placeholder,input[type=radio]:disabled::-ms-input-placeholder{color:#a8bece}.radio-button:disabled::placeholder,input[type=radio]:disabled::placeholder{color:#a8bece}.radio-button:disabled:checked::before,input[type=radio]:disabled:checked:before{background:#e9eff3}.radio-button+span,input[type=radio]+span{display:block;font-weight:400;margin-left:24px}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}select{background:#fff url() no-repeat right 10px center;border-color:#c8d7e1;border-style:solid;border-radius:4px;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:0;outline:0;overflow:hidden;font-size:14px;line-height:21px;font-weight:600;text-overflow:ellipsis;text-decoration:none;vertical-align:top;white-space:nowrap;box-sizing:border-box;padding:7px 32px 9px 14px;-webkit-appearance:none;-moz-appearance:none;appearance:none}select:hover{background-image:url()}select:focus{background-image:url();border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa;outline:0;-moz-outline:none;-moz-user-focus:ignore}select:disabled,select:hover:disabled{background:url() no-repeat right 10px center}select.is-compact{min-width:0;padding:0 20px 2px 6px;margin:0 4px;background-position:right 5px center;background-size:12px 12px}label select,label+select{display:block;min-width:200px}label select.is-compact,label+select.is-compact{display:inline-block;min-width:0}select::-ms-expand{display:none}select::-ms-value{background:0 0;color:#2e4453}select:-moz-focusring{color:transparent;text-shadow:0 0 0 #2e4453}input[type=submit]{padding:0;font-size:14px;-webkit-appearance:none;-moz-appearance:none;appearance:none;vertical-align:baseline;background:#fff;border-color:#c8d7e1;border-style:solid;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:24px 0 0;outline:0;overflow:hidden;font-weight:500;text-overflow:ellipsis;text-decoration:none;vertical-align:top;box-sizing:border-box;font-size:14px;line-height:21px;border-radius:4px;padding:7px 14px 9px;-webkit-appearance:none;-moz-appearance:none;appearance:none}input[type=submit]:hover{border-color:#a8bece;color:#2e4453}input[type=submit]:active{border-width:2px 1px 1px}input[type=submit]:visited{color:#2e4453}input[type=submit]:disabled,input[type=submit][disabled]{color:#e9eff3;background:#fff;border-color:#e9eff3;cursor:default}input[type=submit]:disabled:active,input[type=submit][disabled]:active{border-width:1px 1px 2px}input[type=submit]:focus{border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa}input[type=submit].hidden{display:none}input[type=submit] .gridicon{position:relative;top:4px;margin-top:-2px;width:18px;height:18px}input[type=submit].button-primary{background:#00aadc;border-color:#008ab3;color:#fff}input[type=submit].button-primary:focus,input[type=submit].button-primary:hover{border-color:#005082;color:#fff}input[type=submit].button-primary:disabled,input[type=submit].button-primary[disabled]{background:#bceefd;border-color:#8cc9e2;color:#fff}input[type=submit].button-primary{color:#fff}.ui-sortable-handle{cursor:move}.grunion-section-header{font-size:21px;margin-top:32px;font-weight:600}.grunion-form-settings:hover{box-shadow:0 0 0 1px rgba(200,215,225,.5),0 1px 2px #e9eff3}.grunion-section-header:first-child{margin-top:0}.grunion-type-options{display:flex;flex-wrap:wrap}.grunion-type{flex-grow:0;flex-shrink:0}.grunion-type select{-webkit-appearance:none;width:100%}.grunion-required{padding:27px 0 0 16px;flex-grow:0;flex-shrink:0}.grunion-options{padding-top:16px}.grunion-options ol{list-style:none;padding:0;margin:8px 0 0}.grunion-options li{display:flex;margin-bottom:16px}.grunion-field-edit .grunion-options{display:none}.delete-field,.delete-option{color:#0087be;text-decoration:none;width:40px;line-height:40px;font-size:21px;text-align:center;font-weight:600}.delete-field{position:absolute;top:0;right:0}.grunion-controls{display:flex;flex-wrap:wrap}.grunion-update-controls{text-align:right;flex-grow:1}#add-field{flex-grow:0}.delete-field:before,.delete-option:before{font-family:Dashicons;content:"\f335";display:inline-block;speak:none}.grunion-field-edit.grunion-field-checkbox-multiple .grunion-options,.grunion-field-edit.grunion-field-radio .grunion-options,.grunion-field-edit.grunion-field-select .grunion-options{display:block}.screen-reader-text{position:absolute;margin:-1px;padding:0;height:1px;width:1px;overflow:hidden;clip:rect(0 0 0 0);border:0;word-wrap:normal!important}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/editor-style-rtl.css b/plugins/jetpack/modules/contact-form/css/editor-style-rtl.css new file mode 100644 index 00000000..e6b43f3c --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-style-rtl.css @@ -0,0 +1,613 @@ +/* Do not modify this file directly. It is concatenated from individual module CSS files. */ +/* ========================================================================== +** Normalize +** ======================================================================== */ + +body, +label { + font-family: -apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif; + font-size: 16px; + line-height: 1.4em; +} + +/* ========================================================================== +** Card +** ======================================================================= */ + +.card { + display: block; + position: relative; + margin: 0 auto; + padding: 16px; + box-sizing: border-box; + background: white; + box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3; +} + +.card:after { + content: "."; + display: block; + height: 0; + clear: both; + visibility: hidden; +} + +@media ( min-width: 481px ) { + .card { + padding: 24px; + } +} + +.card > div { + margin-top: 24px; +} + +.card > div:first-child { + margin-top: 0; +} + + +/* ========================================================================== +** Labels +** ======================================================================= */ + +label { + display: block; + font-size: 14px; + font-weight: 600; + margin-bottom: 5px; +} + + +/* ========================================================================== +** Text Inputs +** ======================================================================= */ + +input[type="text"], +input[type="tel"], +input[type="email"], +input[type="url"] { + border-radius: 0; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +input[type="text"]:-ms-input-placeholder, +input[type="tel"]:-ms-input-placeholder, +input[type="email"]:-ms-input-placeholder, +input[type="url"]:-ms-input-placeholder { + color: #87a6bc; +} + +input[type="text"]::-ms-input-placeholder, +input[type="tel"]::-ms-input-placeholder, +input[type="email"]::-ms-input-placeholder, +input[type="url"]::-ms-input-placeholder { + color: #87a6bc; +} + +input[type="text"]::placeholder, +input[type="tel"]::placeholder, +input[type="email"]::placeholder, +input[type="url"]::placeholder { + color: #87a6bc; +} + +input[type="text"]:hover, +input[type="tel"]:hover, +input[type="email"]:hover, +input[type="url"]:hover { + border-color: #a8bece; +} + +input[type="text"]:focus, +input[type="tel"]:focus, +input[type="email"]:focus, +input[type="url"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="text"]:focus::-ms-clear, +input[type="tel"]:focus::-ms-clear, +input[type="email"]:focus::-ms-clear, +input[type="url"]:focus::-ms-clear { + display: none; +} + +input[type="text"]:disabled, +input[type="tel"]:disabled, +input[type="email"]:disabled, +input[type="url"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +input[type="text"]:disabled:hover, +input[type="tel"]:disabled:hover, +input[type="email"]:disabled:hover, +input[type="url"]:disabled:hover { + cursor: default; +} + +input[type="text"]:disabled:-ms-input-placeholder, +input[type="tel"]:disabled:-ms-input-placeholder, +input[type="email"]:disabled:-ms-input-placeholder, +input[type="url"]:disabled:-ms-input-placeholder { + color: #a8bece; +} + +input[type="text"]:disabled::-ms-input-placeholder, +input[type="tel"]:disabled::-ms-input-placeholder, +input[type="email"]:disabled::-ms-input-placeholder, +input[type="url"]:disabled::-ms-input-placeholder { + color: #a8bece; +} + +input[type="text"]:disabled::placeholder, +input[type="tel"]:disabled::placeholder, +input[type="email"]:disabled::placeholder, +input[type="url"]:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Textareas +** ======================================================================= */ + +textarea { + border-radius: 0; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + height: 92px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +textarea:-ms-input-placeholder { + color: #87a6bc; +} + +textarea::-ms-input-placeholder { + color: #87a6bc; +} + +textarea::placeholder { + color: #87a6bc; +} + +textarea:hover { + border-color: #a8bece; +} + +textarea:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +textarea:focus::-ms-clear { + display: none; +} + +textarea:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +textarea:disabled:hover { + cursor: default; +} + +textarea:disabled:-ms-input-placeholder { + color: #a8bece; +} + +textarea:disabled::-ms-input-placeholder { + color: #a8bece; +} + +textarea:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Checkboxes +** ======================================================================= */ + +input[type="checkbox"] { + -webkit-appearance: none; + display: inline-block; + box-sizing: border-box; + margin: 2px 0 0; + padding: 7px 14px; + width: 16px; + height: 16px; + float: right; + outline: 0; + padding: 0; + box-shadow: none; + background-color: #fff; + border: 1px solid #c8d7e1; + color: #2e4453; + font-size: 16px; + line-height: 0; + text-align: center; + vertical-align: middle; + -moz-appearance: none; + appearance: none; + transition: all .15s ease-in-out; + clear: none; + cursor: pointer; +} + +input[type="checkbox"]:checked:before { + content: '\f147'; + font-family: Dashicons; + margin: -3px -4px 0 0; + float: right; + display: inline-block; + vertical-align: middle; + width: 16px; + font-size: 20px; + line-height: 1; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + speak: none; + color: #00aadc; +} + +input[type="checkbox"]:disabled:checked:before { + color: #a8bece; +} + +input[type="checkbox"]:hover { + border-color: #a8bece; +} + +input[type="checkbox"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="checkbox"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; +} + +input[type="checkbox"]:disabled:hover { + cursor: default; +} + +input[type="checkbox"] + span { + display: block; + font-weight: normal; + margin-right: 24px; +} + + +/* ========================================================================== +** Radio buttons +** ======================================================================== */ + +input[type=radio] { + color: #2e4453; + font-size: 16px; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-sizing: border-box; + -webkit-appearance: none; + clear: none; + cursor: pointer; + display: inline-block; + line-height: 0; + height: 16px; + margin: 2px 0 0 4px; + float: right; + outline: 0; + padding: 0; + text-align: center; + vertical-align: middle; + width: 16px; + min-width: 16px; + -moz-appearance: none; + appearance: none; + border-radius: 50%; + line-height: 10px; +} + +input[type="radio"]:hover { + border-color: #a8bece; +} + +input[type="radio"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="radio"]:focus::-ms-clear { + display: none; +} + +input[type="radio"]:checked:before { + float: right; + display: inline-block; + content: '\2022'; + margin: 3px; + width: 8px; + height: 8px; + text-indent: -9999px; + background: #00aadc; + vertical-align: middle; + border-radius: 50%; + animation: grow .2s ease-in-out; +} + +input[type="radio"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; + -webkit-text-fill-color: #a8bece; +} + +input[type="radio"]:disabled:hover { + cursor: default; +} + +input[type="radio"]:disabled:-ms-input-placeholder { + color: #a8bece; +} + +input[type="radio"]:disabled::-ms-input-placeholder { + color: #a8bece; +} + +input[type="radio"]:disabled::placeholder { + color: #a8bece; +} + +input[type="radio"]:disabled:checked:before { + background: #e9eff3; +} + +input[type="radio"] + span { + display: block; + font-weight: normal; + margin-right: 24px; +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + + +/* ========================================================================== +** Selects +** ======================================================================== */ + +select { + background: #fff url() no-repeat left 10px center; + border-color: #c8d7e1; + border-style: solid; + border-radius: 4px; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 0; + outline: 0; + overflow: hidden; + font-size: 14px; + line-height: 21px; + font-weight: 600; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + white-space: nowrap; + box-sizing: border-box; + /* Aligns the text to the 8px baseline grid and adds padding on right to allow for the arrow. */ + padding: 7px 14px 9px 32px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +select:hover { + background-image: url(); +} + +select:focus { + background-image: url(); + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; + outline: 0; + -moz-outline:none; + -moz-user-focus:ignore; +} + +select:disabled, +select:hover:disabled { + background: url() no-repeat left 10px center;; +} + +select.is-compact { + min-width: 0; + padding: 0 6px 2px 20px; + margin: 0 4px; + background-position: left 5px center; + background-size: 12px 12px; +} + +/* Make it display:block when it follows a label */ +label select, +label + select { + display: block; + min-width: 200px; +} + +label select.is-compact, +label + select.is-compact { + display: inline-block; + min-width: 0; +} + +/* IE: Remove the default arrow */ +select::-ms-expand { + display: none; +} + +/* IE: Remove default background and color styles on focus */ +select::-ms-value { + background: none; + color: #2e4453; +} + +/* Firefox: Remove the focus outline, see http://stackoverflow.com/questions/3773430/remove-outline-from-select-box-in-ff/18853002#18853002 */ +select:-moz-focusring { + color: transparent; + text-shadow: 0 0 0 #2e4453; +} + + +/* ========================================================================== +** Buttons +** ======================================================================== */ + +input[type="submit"] { + padding: 0; + font-size: 14px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + vertical-align: baseline; + background: white; + border-color: #c8d7e1; + border-style: solid; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 24px 0 0; + outline: 0; + overflow: hidden; + font-weight: 500; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + box-sizing: border-box; + font-size: 14px; + line-height: 21px; + border-radius: 4px; + padding: 7px 14px 9px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +input[type="submit"]:hover { + border-color: #a8bece; + color: #2e4453; +} + +input[type="submit"]:active { + border-width: 2px 1px 1px; +} + +input[type="submit"]:visited { + color: #2e4453; +} + +input[type="submit"][disabled], +input[type="submit"]:disabled { + color: #e9eff3; + background: white; + border-color: #e9eff3; + cursor: default; +} + +input[type="submit"][disabled]:active, +input[type="submit"]:disabled:active { + border-width: 1px 1px 2px; +} + +input[type="submit"]:focus { + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; +} + +/* ========================================================================== +** Preview styles +** ======================================================================== */ + +.wpview.wpview-wrap[data-wpview-type=contact-form] iframe.inline-edit-contact-form { + width: 100%; + min-height: 500px; + border: 0; + overflow: hidden; + margin-bottom: 0; + display: block; +} + +.contact-submit.contact-submit { + margin-top: 0; + margin-bottom: 0; +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-style-rtl.min.css b/plugins/jetpack/modules/contact-form/css/editor-style-rtl.min.css new file mode 100644 index 00000000..4562abc2 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-style-rtl.min.css @@ -0,0 +1 @@ +body,label{font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif;font-size:16px;line-height:1.4em}.card{display:block;position:relative;margin:0 auto;padding:16px;box-sizing:border-box;background:#fff;box-shadow:0 0 0 1px rgba(200,215,225,.5),0 1px 2px #e9eff3}.card:after{content:".";display:block;height:0;clear:both;visibility:hidden}@media (min-width:481px){.card{padding:24px}}.card>div{margin-top:24px}.card>div:first-child{margin-top:0}label{display:block;font-size:14px;font-weight:600;margin-bottom:5px}input[type=email],input[type=tel],input[type=text],input[type=url]{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}input[type=email]:-ms-input-placeholder,input[type=tel]:-ms-input-placeholder,input[type=text]:-ms-input-placeholder,input[type=url]:-ms-input-placeholder{color:#87a6bc}input[type=email]::-ms-input-placeholder,input[type=tel]::-ms-input-placeholder,input[type=text]::-ms-input-placeholder,input[type=url]::-ms-input-placeholder{color:#87a6bc}input[type=email]::placeholder,input[type=tel]::placeholder,input[type=text]::placeholder,input[type=url]::placeholder{color:#87a6bc}input[type=email]:hover,input[type=tel]:hover,input[type=text]:hover,input[type=url]:hover{border-color:#a8bece}input[type=email]:focus,input[type=tel]:focus,input[type=text]:focus,input[type=url]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=email]:focus::-ms-clear,input[type=tel]:focus::-ms-clear,input[type=text]:focus::-ms-clear,input[type=url]:focus::-ms-clear{display:none}input[type=email]:disabled,input[type=tel]:disabled,input[type=text]:disabled,input[type=url]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}input[type=email]:disabled:hover,input[type=tel]:disabled:hover,input[type=text]:disabled:hover,input[type=url]:disabled:hover{cursor:default}input[type=email]:disabled:-ms-input-placeholder,input[type=tel]:disabled:-ms-input-placeholder,input[type=text]:disabled:-ms-input-placeholder,input[type=url]:disabled:-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::-ms-input-placeholder,input[type=tel]:disabled::-ms-input-placeholder,input[type=text]:disabled::-ms-input-placeholder,input[type=url]:disabled::-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::placeholder,input[type=tel]:disabled::placeholder,input[type=text]:disabled::placeholder,input[type=url]:disabled::placeholder{color:#a8bece}textarea{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;height:92px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}textarea:-ms-input-placeholder{color:#87a6bc}textarea::-ms-input-placeholder{color:#87a6bc}textarea::placeholder{color:#87a6bc}textarea:hover{border-color:#a8bece}textarea:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}textarea:focus::-ms-clear{display:none}textarea:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}textarea:disabled:hover{cursor:default}textarea:disabled:-ms-input-placeholder{color:#a8bece}textarea:disabled::-ms-input-placeholder{color:#a8bece}textarea:disabled::placeholder{color:#a8bece}input[type=checkbox]{-webkit-appearance:none;display:inline-block;box-sizing:border-box;margin:2px 0 0;padding:7px 14px;width:16px;height:16px;float:right;outline:0;padding:0;box-shadow:none;background-color:#fff;border:1px solid #c8d7e1;color:#2e4453;font-size:16px;line-height:0;text-align:center;vertical-align:middle;-moz-appearance:none;appearance:none;transition:all .15s ease-in-out;clear:none;cursor:pointer}input[type=checkbox]:checked:before{content:'\f147';font-family:Dashicons;margin:-3px -4px 0 0;float:right;display:inline-block;vertical-align:middle;width:16px;font-size:20px;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;speak:none;color:#00aadc}input[type=checkbox]:disabled:checked:before{color:#a8bece}input[type=checkbox]:hover{border-color:#a8bece}input[type=checkbox]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=checkbox]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1}input[type=checkbox]:disabled:hover{cursor:default}input[type=checkbox]+span{display:block;font-weight:400;margin-right:24px}input[type=radio]{color:#2e4453;font-size:16px;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-sizing:border-box;-webkit-appearance:none;clear:none;cursor:pointer;display:inline-block;line-height:0;height:16px;margin:2px 0 0 4px;float:right;outline:0;padding:0;text-align:center;vertical-align:middle;width:16px;min-width:16px;-moz-appearance:none;appearance:none;border-radius:50%;line-height:10px}input[type=radio]:hover{border-color:#a8bece}input[type=radio]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=radio]:focus::-ms-clear{display:none}input[type=radio]:checked:before{float:right;display:inline-block;content:'\2022';margin:3px;width:8px;height:8px;text-indent:-9999px;background:#00aadc;vertical-align:middle;border-radius:50%;animation:grow .2s ease-in-out}input[type=radio]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1;-webkit-text-fill-color:#a8bece}input[type=radio]:disabled:hover{cursor:default}input[type=radio]:disabled:-ms-input-placeholder{color:#a8bece}input[type=radio]:disabled::-ms-input-placeholder{color:#a8bece}input[type=radio]:disabled::placeholder{color:#a8bece}input[type=radio]:disabled:checked:before{background:#e9eff3}input[type=radio]+span{display:block;font-weight:400;margin-right:24px}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}select{background:#fff url() no-repeat left 10px center;border-color:#c8d7e1;border-style:solid;border-radius:4px;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:0;outline:0;overflow:hidden;font-size:14px;line-height:21px;font-weight:600;text-overflow:ellipsis;text-decoration:none;vertical-align:top;white-space:nowrap;box-sizing:border-box;padding:7px 14px 9px 32px;-webkit-appearance:none;-moz-appearance:none;appearance:none}select:hover{background-image:url()}select:focus{background-image:url();border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa;outline:0;-moz-outline:none;-moz-user-focus:ignore}select:disabled,select:hover:disabled{background:url() no-repeat left 10px center}select.is-compact{min-width:0;padding:0 6px 2px 20px;margin:0 4px;background-position:left 5px center;background-size:12px 12px}label select,label+select{display:block;min-width:200px}label select.is-compact,label+select.is-compact{display:inline-block;min-width:0}select::-ms-expand{display:none}select::-ms-value{background:0 0;color:#2e4453}select:-moz-focusring{color:transparent;text-shadow:0 0 0 #2e4453}input[type=submit]{padding:0;font-size:14px;-webkit-appearance:none;-moz-appearance:none;appearance:none;vertical-align:baseline;background:#fff;border-color:#c8d7e1;border-style:solid;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:24px 0 0;outline:0;overflow:hidden;font-weight:500;text-overflow:ellipsis;text-decoration:none;vertical-align:top;box-sizing:border-box;font-size:14px;line-height:21px;border-radius:4px;padding:7px 14px 9px;-webkit-appearance:none;-moz-appearance:none;appearance:none}input[type=submit]:hover{border-color:#a8bece;color:#2e4453}input[type=submit]:active{border-width:2px 1px 1px}input[type=submit]:visited{color:#2e4453}input[type=submit]:disabled,input[type=submit][disabled]{color:#e9eff3;background:#fff;border-color:#e9eff3;cursor:default}input[type=submit]:disabled:active,input[type=submit][disabled]:active{border-width:1px 1px 2px}input[type=submit]:focus{border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa}.wpview.wpview-wrap[data-wpview-type=contact-form] iframe.inline-edit-contact-form{width:100%;min-height:500px;border:0;overflow:hidden;margin-bottom:0;display:block}.contact-submit.contact-submit{margin-top:0;margin-bottom:0}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/editor-style.css b/plugins/jetpack/modules/contact-form/css/editor-style.css new file mode 100644 index 00000000..e060f69a --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-style.css @@ -0,0 +1,554 @@ +/* ========================================================================== +** Normalize +** ======================================================================== */ + +body, +label { + font-family: -apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif; + font-size: 16px; + line-height: 1.4em; +} + +/* ========================================================================== +** Card +** ======================================================================= */ + +.card { + display: block; + position: relative; + margin: 0 auto; + padding: 16px; + box-sizing: border-box; + background: white; + box-shadow: 0 0 0 1px rgba(200, 215, 225, 0.5), 0 1px 2px #e9eff3; +} + +.card:after { + content: "."; + display: block; + height: 0; + clear: both; + visibility: hidden; +} + +@media ( min-width: 481px ) { + .card { + padding: 24px; + } +} + +.card > div { + margin-top: 24px; +} + +.card > div:first-child { + margin-top: 0; +} + + +/* ========================================================================== +** Labels +** ======================================================================= */ + +label { + display: block; + font-size: 14px; + font-weight: 600; + margin-bottom: 5px; +} + + +/* ========================================================================== +** Text Inputs +** ======================================================================= */ + +input[type="text"], +input[type="tel"], +input[type="email"], +input[type="url"] { + border-radius: 0; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +input[type="text"]::placeholder, +input[type="tel"]::placeholder, +input[type="email"]::placeholder, +input[type="url"]::placeholder { + color: #87a6bc; +} + +input[type="text"]:hover, +input[type="tel"]:hover, +input[type="email"]:hover, +input[type="url"]:hover { + border-color: #a8bece; +} + +input[type="text"]:focus, +input[type="tel"]:focus, +input[type="email"]:focus, +input[type="url"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="text"]:focus::-ms-clear, +input[type="tel"]:focus::-ms-clear, +input[type="email"]:focus::-ms-clear, +input[type="url"]:focus::-ms-clear { + display: none; +} + +input[type="text"]:disabled, +input[type="tel"]:disabled, +input[type="email"]:disabled, +input[type="url"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +input[type="text"]:disabled:hover, +input[type="tel"]:disabled:hover, +input[type="email"]:disabled:hover, +input[type="url"]:disabled:hover { + cursor: default; +} + +input[type="text"]:disabled::placeholder, +input[type="tel"]:disabled::placeholder, +input[type="email"]:disabled::placeholder, +input[type="url"]:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Textareas +** ======================================================================= */ + +textarea { + border-radius: 0; + appearance: none; + box-sizing: border-box; + margin: 0; + padding: 7px 14px; + height: 92px; + width: 100%; + color: #2e4453; + font-size: 16px; + line-height: 1.5; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-shadow: none; +} + +textarea::placeholder { + color: #87a6bc; +} + +textarea:hover { + border-color: #a8bece; +} + +textarea:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +textarea:focus::-ms-clear { + display: none; +} + +textarea:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + -webkit-text-fill-color: #a8bece; +} + +textarea:disabled:hover { + cursor: default; +} + +textarea:disabled::placeholder { + color: #a8bece; +} + + +/* ========================================================================== +** Checkboxes +** ======================================================================= */ + +input[type="checkbox"] { + -webkit-appearance: none; + display: inline-block; + box-sizing: border-box; + margin: 2px 0 0; + padding: 7px 14px; + width: 16px; + height: 16px; + float: left; + outline: 0; + padding: 0; + box-shadow: none; + background-color: #fff; + border: 1px solid #c8d7e1; + color: #2e4453; + font-size: 16px; + line-height: 0; + text-align: center; + vertical-align: middle; + appearance: none; + transition: all .15s ease-in-out; + clear: none; + cursor: pointer; +} + +input[type="checkbox"]:checked:before { + content: '\f147'; + font-family: Dashicons; + margin: -3px 0 0 -4px; + float: left; + display: inline-block; + vertical-align: middle; + width: 16px; + font-size: 20px; + line-height: 1; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + speak: none; + color: #00aadc; +} + +input[type="checkbox"]:disabled:checked:before { + color: #a8bece; +} + +input[type="checkbox"]:hover { + border-color: #a8bece; +} + +input[type="checkbox"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="checkbox"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; +} + +input[type="checkbox"]:disabled:hover { + cursor: default; +} + +input[type="checkbox"] + span { + display: block; + font-weight: normal; + margin-left: 24px; +} + + +/* ========================================================================== +** Radio buttons +** ======================================================================== */ + +input[type=radio] { + color: #2e4453; + font-size: 16px; + border: 1px solid #c8d7e1; + background-color: #fff; + transition: all .15s ease-in-out; + box-sizing: border-box; + -webkit-appearance: none; + clear: none; + cursor: pointer; + display: inline-block; + line-height: 0; + height: 16px; + margin: 2px 4px 0 0; + float: left; + outline: 0; + padding: 0; + text-align: center; + vertical-align: middle; + width: 16px; + min-width: 16px; + appearance: none; + border-radius: 50%; + line-height: 10px; +} + +input[type="radio"]:hover { + border-color: #a8bece; +} + +input[type="radio"]:focus { + border-color: #0087be; + outline: none; + box-shadow: 0 0 0 2px #78dcfa; +} + +input[type="radio"]:focus::-ms-clear { + display: none; +} + +input[type="radio"]:checked:before { + float: left; + display: inline-block; + content: '\2022'; + margin: 3px; + width: 8px; + height: 8px; + text-indent: -9999px; + background: #00aadc; + vertical-align: middle; + border-radius: 50%; + animation: grow .2s ease-in-out; +} + +input[type="radio"]:disabled { + background: #f3f6f8; + border-color: #e9eff3; + color: #a8bece; + opacity: 1; + -webkit-text-fill-color: #a8bece; +} + +input[type="radio"]:disabled:hover { + cursor: default; +} + +input[type="radio"]:disabled::placeholder { + color: #a8bece; +} + +input[type="radio"]:disabled:checked:before { + background: #e9eff3; +} + +input[type="radio"] + span { + display: block; + font-weight: normal; + margin-left: 24px; +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + +@keyframes grow { + 0% { + transform: scale(0.3); + } + + 60% { + transform: scale(1.15); + } + + 100% { + transform: scale(1); + } +} + + +/* ========================================================================== +** Selects +** ======================================================================== */ + +select { + background: #fff url() no-repeat right 10px center; + border-color: #c8d7e1; + border-style: solid; + border-radius: 4px; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 0; + outline: 0; + overflow: hidden; + font-size: 14px; + line-height: 21px; + font-weight: 600; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + white-space: nowrap; + box-sizing: border-box; + /* Aligns the text to the 8px baseline grid and adds padding on right to allow for the arrow. */ + padding: 7px 32px 9px 14px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +select:hover { + background-image: url(); +} + +select:focus { + background-image: url(); + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; + outline: 0; + -moz-outline:none; + -moz-user-focus:ignore; +} + +select:disabled, +select:hover:disabled { + background: url() no-repeat right 10px center;; +} + +select.is-compact { + min-width: 0; + padding: 0 20px 2px 6px; + margin: 0 4px; + background-position: right 5px center; + background-size: 12px 12px; +} + +/* Make it display:block when it follows a label */ +label select, +label + select { + display: block; + min-width: 200px; +} + +label select.is-compact, +label + select.is-compact { + display: inline-block; + min-width: 0; +} + +/* IE: Remove the default arrow */ +select::-ms-expand { + display: none; +} + +/* IE: Remove default background and color styles on focus */ +select::-ms-value { + background: none; + color: #2e4453; +} + +/* Firefox: Remove the focus outline, see http://stackoverflow.com/questions/3773430/remove-outline-from-select-box-in-ff/18853002#18853002 */ +select:-moz-focusring { + color: transparent; + text-shadow: 0 0 0 #2e4453; +} + + +/* ========================================================================== +** Buttons +** ======================================================================== */ + +input[type="submit"] { + padding: 0; + font-size: 14px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + vertical-align: baseline; + background: white; + border-color: #c8d7e1; + border-style: solid; + border-width: 1px 1px 2px; + color: #2e4453; + cursor: pointer; + display: inline-block; + margin: 24px 0 0; + outline: 0; + overflow: hidden; + font-weight: 500; + text-overflow: ellipsis; + text-decoration: none; + vertical-align: top; + box-sizing: border-box; + font-size: 14px; + line-height: 21px; + border-radius: 4px; + padding: 7px 14px 9px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; +} + +input[type="submit"]:hover { + border-color: #a8bece; + color: #2e4453; +} + +input[type="submit"]:active { + border-width: 2px 1px 1px; +} + +input[type="submit"]:visited { + color: #2e4453; +} + +input[type="submit"][disabled], +input[type="submit"]:disabled { + color: #e9eff3; + background: white; + border-color: #e9eff3; + cursor: default; +} + +input[type="submit"][disabled]:active, +input[type="submit"]:disabled:active { + border-width: 1px 1px 2px; +} + +input[type="submit"]:focus { + border-color: #00aadc; + box-shadow: 0 0 0 2px #78dcfa; +} + +/* ========================================================================== +** Preview styles +** ======================================================================== */ + +.wpview.wpview-wrap[data-wpview-type=contact-form] iframe.inline-edit-contact-form { + width: 100%; + min-height: 500px; + border: 0; + overflow: hidden; + margin-bottom: 0; + display: block; +} + +.contact-submit.contact-submit { + margin-top: 0; + margin-bottom: 0; +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-style.min.css b/plugins/jetpack/modules/contact-form/css/editor-style.min.css new file mode 100644 index 00000000..fdef6b83 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-style.min.css @@ -0,0 +1,2 @@ +/* Do not modify this file directly. It is concatenated from individual module CSS files. */ +body,label{font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",sans-serif;font-size:16px;line-height:1.4em}.card{display:block;position:relative;margin:0 auto;padding:16px;box-sizing:border-box;background:#fff;box-shadow:0 0 0 1px rgba(200,215,225,.5),0 1px 2px #e9eff3}.card:after{content:".";display:block;height:0;clear:both;visibility:hidden}@media (min-width:481px){.card{padding:24px}}.card>div{margin-top:24px}.card>div:first-child{margin-top:0}label{display:block;font-size:14px;font-weight:600;margin-bottom:5px}input[type=email],input[type=tel],input[type=text],input[type=url]{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}input[type=email]:-ms-input-placeholder,input[type=tel]:-ms-input-placeholder,input[type=text]:-ms-input-placeholder,input[type=url]:-ms-input-placeholder{color:#87a6bc}input[type=email]::-ms-input-placeholder,input[type=tel]::-ms-input-placeholder,input[type=text]::-ms-input-placeholder,input[type=url]::-ms-input-placeholder{color:#87a6bc}input[type=email]::placeholder,input[type=tel]::placeholder,input[type=text]::placeholder,input[type=url]::placeholder{color:#87a6bc}input[type=email]:hover,input[type=tel]:hover,input[type=text]:hover,input[type=url]:hover{border-color:#a8bece}input[type=email]:focus,input[type=tel]:focus,input[type=text]:focus,input[type=url]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=email]:focus::-ms-clear,input[type=tel]:focus::-ms-clear,input[type=text]:focus::-ms-clear,input[type=url]:focus::-ms-clear{display:none}input[type=email]:disabled,input[type=tel]:disabled,input[type=text]:disabled,input[type=url]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}input[type=email]:disabled:hover,input[type=tel]:disabled:hover,input[type=text]:disabled:hover,input[type=url]:disabled:hover{cursor:default}input[type=email]:disabled:-ms-input-placeholder,input[type=tel]:disabled:-ms-input-placeholder,input[type=text]:disabled:-ms-input-placeholder,input[type=url]:disabled:-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::-ms-input-placeholder,input[type=tel]:disabled::-ms-input-placeholder,input[type=text]:disabled::-ms-input-placeholder,input[type=url]:disabled::-ms-input-placeholder{color:#a8bece}input[type=email]:disabled::placeholder,input[type=tel]:disabled::placeholder,input[type=text]:disabled::placeholder,input[type=url]:disabled::placeholder{color:#a8bece}textarea{border-radius:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;box-sizing:border-box;margin:0;padding:7px 14px;height:92px;width:100%;color:#2e4453;font-size:16px;line-height:1.5;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-shadow:none}textarea:-ms-input-placeholder{color:#87a6bc}textarea::-ms-input-placeholder{color:#87a6bc}textarea::placeholder{color:#87a6bc}textarea:hover{border-color:#a8bece}textarea:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}textarea:focus::-ms-clear{display:none}textarea:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;-webkit-text-fill-color:#a8bece}textarea:disabled:hover{cursor:default}textarea:disabled:-ms-input-placeholder{color:#a8bece}textarea:disabled::-ms-input-placeholder{color:#a8bece}textarea:disabled::placeholder{color:#a8bece}input[type=checkbox]{-webkit-appearance:none;display:inline-block;box-sizing:border-box;margin:2px 0 0;padding:7px 14px;width:16px;height:16px;float:left;outline:0;padding:0;box-shadow:none;background-color:#fff;border:1px solid #c8d7e1;color:#2e4453;font-size:16px;line-height:0;text-align:center;vertical-align:middle;-moz-appearance:none;appearance:none;transition:all .15s ease-in-out;clear:none;cursor:pointer}input[type=checkbox]:checked:before{content:'\f147';font-family:Dashicons;margin:-3px 0 0 -4px;float:left;display:inline-block;vertical-align:middle;width:16px;font-size:20px;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;speak:none;color:#00aadc}input[type=checkbox]:disabled:checked:before{color:#a8bece}input[type=checkbox]:hover{border-color:#a8bece}input[type=checkbox]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=checkbox]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1}input[type=checkbox]:disabled:hover{cursor:default}input[type=checkbox]+span{display:block;font-weight:400;margin-left:24px}input[type=radio]{color:#2e4453;font-size:16px;border:1px solid #c8d7e1;background-color:#fff;transition:all .15s ease-in-out;box-sizing:border-box;-webkit-appearance:none;clear:none;cursor:pointer;display:inline-block;line-height:0;height:16px;margin:2px 4px 0 0;float:left;outline:0;padding:0;text-align:center;vertical-align:middle;width:16px;min-width:16px;-moz-appearance:none;appearance:none;border-radius:50%;line-height:10px}input[type=radio]:hover{border-color:#a8bece}input[type=radio]:focus{border-color:#0087be;outline:0;box-shadow:0 0 0 2px #78dcfa}input[type=radio]:focus::-ms-clear{display:none}input[type=radio]:checked:before{float:left;display:inline-block;content:'\2022';margin:3px;width:8px;height:8px;text-indent:-9999px;background:#00aadc;vertical-align:middle;border-radius:50%;animation:grow .2s ease-in-out}input[type=radio]:disabled{background:#f3f6f8;border-color:#e9eff3;color:#a8bece;opacity:1;-webkit-text-fill-color:#a8bece}input[type=radio]:disabled:hover{cursor:default}input[type=radio]:disabled:-ms-input-placeholder{color:#a8bece}input[type=radio]:disabled::-ms-input-placeholder{color:#a8bece}input[type=radio]:disabled::placeholder{color:#a8bece}input[type=radio]:disabled:checked:before{background:#e9eff3}input[type=radio]+span{display:block;font-weight:400;margin-left:24px}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}@keyframes grow{0%{transform:scale(.3)}60%{transform:scale(1.15)}100%{transform:scale(1)}}select{background:#fff url() no-repeat right 10px center;border-color:#c8d7e1;border-style:solid;border-radius:4px;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:0;outline:0;overflow:hidden;font-size:14px;line-height:21px;font-weight:600;text-overflow:ellipsis;text-decoration:none;vertical-align:top;white-space:nowrap;box-sizing:border-box;padding:7px 32px 9px 14px;-webkit-appearance:none;-moz-appearance:none;appearance:none}select:hover{background-image:url()}select:focus{background-image:url();border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa;outline:0;-moz-outline:none;-moz-user-focus:ignore}select:disabled,select:hover:disabled{background:url() no-repeat right 10px center}select.is-compact{min-width:0;padding:0 20px 2px 6px;margin:0 4px;background-position:right 5px center;background-size:12px 12px}label select,label+select{display:block;min-width:200px}label select.is-compact,label+select.is-compact{display:inline-block;min-width:0}select::-ms-expand{display:none}select::-ms-value{background:0 0;color:#2e4453}select:-moz-focusring{color:transparent;text-shadow:0 0 0 #2e4453}input[type=submit]{padding:0;font-size:14px;-webkit-appearance:none;-moz-appearance:none;appearance:none;vertical-align:baseline;background:#fff;border-color:#c8d7e1;border-style:solid;border-width:1px 1px 2px;color:#2e4453;cursor:pointer;display:inline-block;margin:24px 0 0;outline:0;overflow:hidden;font-weight:500;text-overflow:ellipsis;text-decoration:none;vertical-align:top;box-sizing:border-box;font-size:14px;line-height:21px;border-radius:4px;padding:7px 14px 9px;-webkit-appearance:none;-moz-appearance:none;appearance:none}input[type=submit]:hover{border-color:#a8bece;color:#2e4453}input[type=submit]:active{border-width:2px 1px 1px}input[type=submit]:visited{color:#2e4453}input[type=submit]:disabled,input[type=submit][disabled]{color:#e9eff3;background:#fff;border-color:#e9eff3;cursor:default}input[type=submit]:disabled:active,input[type=submit][disabled]:active{border-width:1px 1px 2px}input[type=submit]:focus{border-color:#00aadc;box-shadow:0 0 0 2px #78dcfa}.wpview.wpview-wrap[data-wpview-type=contact-form] iframe.inline-edit-contact-form{width:100%;min-height:500px;border:0;overflow:hidden;margin-bottom:0;display:block}.contact-submit.contact-submit{margin-top:0;margin-bottom:0}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/editor-ui-rtl.css b/plugins/jetpack/modules/contact-form/css/editor-ui-rtl.css new file mode 100644 index 00000000..df4c059d --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-ui-rtl.css @@ -0,0 +1,27 @@ +/* Do not modify this file directly. It is concatenated from individual module CSS files. */ +i.mce-i-grunion { + font-size: 20px; +} + +i.mce-i-grunion:before, +.jetpack-contact-form-icon:before { + width: 24px; + vertical-align: top; + content: ''; + display: block; + height: 24px; + background-size: 24px; + background-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path fill="rgb(85, 93, 102)" d="M13 7.5h5v2h-5zm0 7h5v2h-5zM19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14zM11 6H6v5h5V6zm-1 4H7V7h3v3zm1 3H6v5h5v-5zm-1 4H7v-3h3v3z"/></svg>'); + margin-top: -4px; +} +i.mce-i-grunion:before { + margin-top: -2px; + margin-right: -2px; +} + +.jetpack-contact-form-icon { + opacity: 0.7; + vertical-align: text-top; + display: inline-block; + height: 18px; +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-ui-rtl.min.css b/plugins/jetpack/modules/contact-form/css/editor-ui-rtl.min.css new file mode 100644 index 00000000..3d7d4a17 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-ui-rtl.min.css @@ -0,0 +1 @@ +i.mce-i-grunion{font-size:20px}.jetpack-contact-form-icon:before,i.mce-i-grunion:before{width:24px;vertical-align:top;content:'';display:block;height:24px;background-size:24px;background-image:url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path fill="rgb(85, 93, 102)" d="M13 7.5h5v2h-5zm0 7h5v2h-5zM19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14zM11 6H6v5h5V6zm-1 4H7V7h3v3zm1 3H6v5h5v-5zm-1 4H7v-3h3v3z"/></svg>');margin-top:-4px}i.mce-i-grunion:before{margin-top:-2px;margin-right:-2px}.jetpack-contact-form-icon{opacity:.7;vertical-align:text-top;display:inline-block;height:18px}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/editor-ui.css b/plugins/jetpack/modules/contact-form/css/editor-ui.css new file mode 100644 index 00000000..b68f810b --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-ui.css @@ -0,0 +1,26 @@ +i.mce-i-grunion { + font-size: 20px; +} + +i.mce-i-grunion:before, +.jetpack-contact-form-icon:before { + width: 24px; + vertical-align: top; + content: ''; + display: block; + height: 24px; + background-size: 24px; + background-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path fill="rgb(85, 93, 102)" d="M13 7.5h5v2h-5zm0 7h5v2h-5zM19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14zM11 6H6v5h5V6zm-1 4H7V7h3v3zm1 3H6v5h5v-5zm-1 4H7v-3h3v3z"/></svg>'); + margin-top: -4px; +} +i.mce-i-grunion:before { + margin-top: -2px; + margin-left: -2px; +} + +.jetpack-contact-form-icon { + opacity: 0.7; + vertical-align: text-top; + display: inline-block; + height: 18px; +} diff --git a/plugins/jetpack/modules/contact-form/css/editor-ui.min.css b/plugins/jetpack/modules/contact-form/css/editor-ui.min.css new file mode 100644 index 00000000..f7c96dfb --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/editor-ui.min.css @@ -0,0 +1,2 @@ +/* Do not modify this file directly. It is concatenated from individual module CSS files. */ +i.mce-i-grunion{font-size:20px}.jetpack-contact-form-icon:before,i.mce-i-grunion:before{width:24px;vertical-align:top;content:'';display:block;height:24px;background-size:24px;background-image:url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path fill="rgb(85, 93, 102)" d="M13 7.5h5v2h-5zm0 7h5v2h-5zM19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14zM11 6H6v5h5V6zm-1 4H7V7h3v3zm1 3H6v5h5v-5zm-1 4H7v-3h3v3z"/></svg>');margin-top:-4px}i.mce-i-grunion:before{margin-top:-2px;margin-left:-2px}.jetpack-contact-form-icon{opacity:.7;vertical-align:text-top;display:inline-block;height:18px}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/grunion-rtl.css b/plugins/jetpack/modules/contact-form/css/grunion-rtl.css new file mode 100644 index 00000000..952b25fa --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/grunion-rtl.css @@ -0,0 +1 @@ +.contact-form .clear-form{clear:both}.contact-form input:-ms-input-placeholder{transition:opacity .3s ease-out}.contact-form input::-ms-input-placeholder{transition:opacity .3s ease-out}.contact-form input::placeholder{transition:opacity .3s ease-out}.contact-form input:hover:-ms-input-placeholder{opacity:.5}.contact-form input:hover::-ms-input-placeholder{opacity:.5}.contact-form input:hover::placeholder{opacity:.5}.contact-form input:focus:-ms-input-placeholder{opacity:.3}.contact-form input:focus::-ms-input-placeholder{opacity:.3}.contact-form input:focus::placeholder{opacity:.3}.contact-form input[type=email],.contact-form input[type=text],.contact-form input[type=url]{width:300px;max-width:98%;margin-bottom:13px}.contact-form select{margin-bottom:13px}.contact-form textarea{height:200px;width:80%;float:none;margin-bottom:13px}.contact-form input[type=checkbox],.contact-form input[type=radio]{float:none;margin-bottom:13px}.contact-form label{margin-bottom:3px;float:none;font-weight:700;display:block}.contact-form label.checkbox,.contact-form label.radio{margin-bottom:3px;float:none;font-weight:700;display:inline-block}.contact-form label span{color:#aaa;margin-right:4px;font-weight:400}.contact-form-submission{margin-bottom:4em;padding:1.5em 1em}.contact-form-submission p{margin:0 auto;word-wrap:break-word}.form-errors .form-error-message{color:red}.textwidget .contact-form input[type=email],.textwidget .contact-form input[type=text],.textwidget .contact-form input[type=url],.textwidget .contact-form textarea{width:250px;max-width:100%;box-sizing:border-box}#jetpack-check-feedback-spam{margin:1px 0 0 8px}.jetpack-check-feedback-spam-spinner{display:inline-block;margin-top:7px}
\ No newline at end of file diff --git a/plugins/jetpack/modules/contact-form/css/grunion.css b/plugins/jetpack/modules/contact-form/css/grunion.css new file mode 100644 index 00000000..1089188d --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/grunion.css @@ -0,0 +1,91 @@ +.contact-form .clear-form { + clear: both; +} + +.contact-form input::placeholder { + transition: opacity .3s ease-out; +} +.contact-form input:hover::placeholder { + opacity: 0.5; +} +.contact-form input:focus::placeholder { + opacity: 0.3; +} + +.contact-form input[type='text'], +.contact-form input[type='email'], +.contact-form input[type='url'] { + width: 300px; + max-width: 98%; + margin-bottom: 13px; +} + +.contact-form select { + margin-bottom: 13px; +} + +.contact-form textarea { + height: 200px; + width: 80%; + float: none; + margin-bottom: 13px; +} + +.contact-form input[type='radio'], +.contact-form input[type='checkbox'] { + float: none; + margin-bottom: 13px; +} + +.contact-form label { + margin-bottom: 3px; + float: none; + font-weight: bold; + display: block; +} + +.contact-form label.checkbox, +.contact-form label.radio { + margin-bottom: 3px; + float: none; + font-weight: bold; + display: inline-block; +} + +.contact-form label span { + color: #AAA; + margin-left: 4px; + font-weight: normal; +} + +.contact-form-submission { + margin-bottom: 4em; + padding: 1.5em 1em; +} + +.contact-form-submission p { + margin: 0 auto; + word-wrap: break-word; +} + +.form-errors .form-error-message { + color: red; +} + +.textwidget .contact-form input[type='text'], +.textwidget .contact-form input[type='email'], +.textwidget .contact-form input[type='url'], +.textwidget .contact-form textarea { + width: 250px; + max-width: 100%; + box-sizing: border-box; +} + +#jetpack-check-feedback-spam { + margin: 1px 8px 0px 0px; +} + +.jetpack-check-feedback-spam-spinner { + display: inline-block; + margin-top: 7px; +} diff --git a/plugins/jetpack/modules/contact-form/css/jquery-ui-datepicker.css b/plugins/jetpack/modules/contact-form/css/jquery-ui-datepicker.css new file mode 100644 index 00000000..e82283eb --- /dev/null +++ b/plugins/jetpack/modules/contact-form/css/jquery-ui-datepicker.css @@ -0,0 +1,160 @@ +.ui-datepicker { + padding: 0; + margin: 0; + -webkit-border-radius: 0; + -moz-border-radius: 0; + border-radius: 0; + background-color: #fff; + border: 1px solid #dfdfdf; + border-top: none; + -webkit-box-shadow: 0 3px 6px rgba(0, 0, 0, 0.075); + box-shadow: 0 3px 6px rgba(0, 0, 0, 0.075); + width: auto; +} + +.ui-datepicker * { + padding: 0; + -webkit-border-radius: 0; + -moz-border-radius: 0; + border-radius: 0; +} + +.ui-datepicker table { + width: auto; + margin: 0; + border: none; + border-collapse: collapse; +} + +.ui-datepicker .ui-widget-header, +.ui-datepicker .ui-datepicker-header { + background-image: none; + border: none; + font-weight: normal; +} + +.ui-datepicker .ui-datepicker-header .ui-state-hover { + background: transparent; + border-color: transparent; + cursor: pointer; +} + +.ui-datepicker .ui-datepicker-title { + margin: 0; + padding: 10px 0; + font-size: 14px; + line-height: 14px; + text-align: center; +} + +.ui-datepicker .ui-datepicker-prev, +.ui-datepicker .ui-datepicker-next { + position: relative; + top: 0; + height: 34px; + width: 34px; +} + +.ui-datepicker .ui-state-hover.ui-datepicker-prev, +.ui-datepicker .ui-state-hover.ui-datepicker-next { + border: none; +} + +.ui-datepicker .ui-datepicker-prev, +.ui-datepicker .ui-datepicker-prev-hover { + left: 0; +} + +.ui-datepicker .ui-datepicker-next, +.ui-datepicker .ui-datepicker-next-hover { + right: 0; +} + +.ui-datepicker .ui-datepicker-next span, +.ui-datepicker .ui-datepicker-prev span { + display: none; +} + +.ui-datepicker .ui-datepicker-prev { + float: left; +} + +.ui-datepicker .ui-datepicker-next { + float: right; +} + +.ui-datepicker .ui-datepicker-prev:before, +.ui-datepicker .ui-datepicker-next:before { + font: normal 20px/34px 'dashicons'; + padding-left: 7px; + speak: none; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + width: 34px; + height: 34px; +} + +.ui-datepicker .ui-datepicker-prev:before { + content: '\f341'; +} + +.ui-datepicker .ui-datepicker-next:before { + content: '\f345'; +} + +.ui-datepicker .ui-datepicker-prev-hover:before, +.ui-datepicker .ui-datepicker-next-hover:before { + opacity: 0.7; +} + +.ui-datepicker select.ui-datepicker-month, +.ui-datepicker select.ui-datepicker-year { + width: 33%; +} + +.ui-datepicker thead { + font-weight: 600; +} + +.ui-datepicker th { + padding: 10px; + border-width: 1px; +} + +.ui-datepicker td { + padding: 0; + border: 1px solid #f4f4f4; +} + +.ui-datepicker td.ui-datepicker-other-month { + border: transparent; +} + +.ui-datepicker td.ui-datepicker-week-end { + background-color: #f4f4f4; + border: 1px solid #f4f4f4; +} + +.ui-datepicker td.ui-datepicker-today { + background-color: #f0f0c0; +} + +.ui-datepicker td.ui-datepicker-current-day { + background: #bbdd88; +} + +.ui-datepicker td .ui-state-default { + background: transparent; + border: none; + text-align: center; + text-decoration: none; + width: auto; + display: block; + padding: 5px 10px; + font-weight: normal; + color: #444; +} + +.ui-datepicker td.ui-state-disabled .ui-state-default { + opacity: 0.5; +} diff --git a/plugins/jetpack/modules/contact-form/grunion-contact-form.php b/plugins/jetpack/modules/contact-form/grunion-contact-form.php new file mode 100644 index 00000000..a26dc20c --- /dev/null +++ b/plugins/jetpack/modules/contact-form/grunion-contact-form.php @@ -0,0 +1,3501 @@ +<?php + +/* +Plugin Name: Grunion Contact Form +Description: Add a contact form to any post, page or text widget. Emails will be sent to the post's author by default, or any email address you choose. As seen on WordPress.com. +Plugin URI: http://automattic.com/# +AUthor: Automattic, Inc. +Author URI: http://automattic.com/ +Version: 2.4 +License: GPLv2 or later +*/ + +define( 'GRUNION_PLUGIN_DIR', plugin_dir_path( __FILE__ ) ); +define( 'GRUNION_PLUGIN_URL', plugin_dir_url( __FILE__ ) ); + +if ( is_admin() ) { + require_once GRUNION_PLUGIN_DIR . 'admin.php'; +} + +add_action( 'rest_api_init', 'grunion_contact_form_require_endpoint' ); +function grunion_contact_form_require_endpoint() { + require_once GRUNION_PLUGIN_DIR . 'class-grunion-contact-form-endpoint.php'; +} + +/** + * Sets up various actions, filters, post types, post statuses, shortcodes. + */ +class Grunion_Contact_Form_Plugin { + + /** + * @var string The Widget ID of the widget currently being processed. Used to build the unique contact-form ID for forms embedded in widgets. + */ + public $current_widget_id; + + static $using_contact_form_field = false; + + /** + * @var int The last Feedback Post ID Erased as part of the Personal Data Eraser. + * Helps with pagination. + */ + private $pde_last_post_id_erased = 0; + + /** + * @var string The email address for which we are deleting/exporting all feedbacks + * as part of a Personal Data Eraser or Personal Data Exporter request. + */ + private $pde_email_address = ''; + + static function init() { + static $instance = false; + + if ( ! $instance ) { + $instance = new Grunion_Contact_Form_Plugin; + + // Schedule our daily cleanup + add_action( 'wp_scheduled_delete', array( $instance, 'daily_akismet_meta_cleanup' ) ); + } + + return $instance; + } + + /** + * Runs daily to clean up spam detection metadata after 15 days. Keeps your DB squeaky clean. + */ + public function daily_akismet_meta_cleanup() { + global $wpdb; + + $feedback_ids = $wpdb->get_col( "SELECT p.ID FROM {$wpdb->posts} as p INNER JOIN {$wpdb->postmeta} as m on m.post_id = p.ID WHERE p.post_type = 'feedback' AND m.meta_key = '_feedback_akismet_values' AND DATE_SUB(NOW(), INTERVAL 15 DAY) > p.post_date_gmt LIMIT 10000" ); + + if ( empty( $feedback_ids ) ) { + return; + } + + /** + * Fires right before deleting the _feedback_akismet_values post meta on $feedback_ids + * + * @module contact-form + * + * @since 6.1.0 + * + * @param array $feedback_ids list of feedback post ID + */ + do_action( 'jetpack_daily_akismet_meta_cleanup_before', $feedback_ids ); + foreach ( $feedback_ids as $feedback_id ) { + delete_post_meta( $feedback_id, '_feedback_akismet_values' ); + } + + /** + * Fires right after deleting the _feedback_akismet_values post meta on $feedback_ids + * + * @module contact-form + * + * @since 6.1.0 + * + * @param array $feedback_ids list of feedback post ID + */ + do_action( 'jetpack_daily_akismet_meta_cleanup_after', $feedback_ids ); + } + + /** + * Strips HTML tags from input. Output is NOT HTML safe. + * + * @param mixed $data_with_tags + * @return mixed + */ + public static function strip_tags( $data_with_tags ) { + if ( is_array( $data_with_tags ) ) { + foreach ( $data_with_tags as $index => $value ) { + $index = sanitize_text_field( strval( $index ) ); + $value = wp_kses( strval( $value ), array() ); + $value = str_replace( '&', '&', $value ); // undo damage done by wp_kses_normalize_entities() + + $data_without_tags[ $index ] = $value; + } + } else { + $data_without_tags = wp_kses( $data_with_tags, array() ); + $data_without_tags = str_replace( '&', '&', $data_without_tags ); // undo damage done by wp_kses_normalize_entities() + } + + return $data_without_tags; + } + + /** + * Class uses singleton pattern; use Grunion_Contact_Form_Plugin::init() to initialize. + */ + protected function __construct() { + $this->add_shortcode(); + + // While generating the output of a text widget with a contact-form shortcode, we need to know its widget ID. + add_action( 'dynamic_sidebar', array( $this, 'track_current_widget' ) ); + + // Add a "widget" shortcode attribute to all contact-form shortcodes embedded in widgets + add_filter( 'widget_text', array( $this, 'widget_atts' ), 0 ); + + // If Text Widgets don't get shortcode processed, hack ours into place. + if ( + version_compare( get_bloginfo( 'version' ), '4.9-z', '<=' ) + && ! has_filter( 'widget_text', 'do_shortcode' ) + ) { + add_filter( 'widget_text', array( $this, 'widget_shortcode_hack' ), 5 ); + } + + add_filter( 'jetpack_contact_form_is_spam', array( $this, 'is_spam_blacklist' ), 10, 2 ); + + // Akismet to the rescue + if ( defined( 'AKISMET_VERSION' ) || function_exists( 'akismet_http_post' ) ) { + add_filter( 'jetpack_contact_form_is_spam', array( $this, 'is_spam_akismet' ), 10, 2 ); + add_action( 'contact_form_akismet', array( $this, 'akismet_submit' ), 10, 2 ); + } + + add_action( 'loop_start', array( 'Grunion_Contact_Form', '_style_on' ) ); + + add_action( 'wp_ajax_grunion-contact-form', array( $this, 'ajax_request' ) ); + add_action( 'wp_ajax_nopriv_grunion-contact-form', array( $this, 'ajax_request' ) ); + + // GDPR: personal data exporter & eraser. + add_filter( 'wp_privacy_personal_data_exporters', array( $this, 'register_personal_data_exporter' ) ); + add_filter( 'wp_privacy_personal_data_erasers', array( $this, 'register_personal_data_eraser' ) ); + + // Export to CSV feature + if ( is_admin() ) { + add_action( 'admin_init', array( $this, 'download_feedback_as_csv' ) ); + add_action( 'admin_footer-edit.php', array( $this, 'export_form' ) ); + add_action( 'admin_menu', array( $this, 'admin_menu' ) ); + add_action( 'current_screen', array( $this, 'unread_count' ) ); + } + + // custom post type we'll use to keep copies of the feedback items + register_post_type( + 'feedback', array( + 'labels' => array( + 'name' => __( 'Feedback', 'jetpack' ), + 'singular_name' => __( 'Feedback', 'jetpack' ), + 'search_items' => __( 'Search Feedback', 'jetpack' ), + 'not_found' => __( 'No feedback found', 'jetpack' ), + 'not_found_in_trash' => __( 'No feedback found', 'jetpack' ), + ), + // Matrial Ballot icon + 'menu_icon' => 'data:image/svg+xml;base64,' . base64_encode('<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path fill="none" d="M13 7.5h5v2h-5zm0 7h5v2h-5zM19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V5h14v14zM11 6H6v5h5V6zm-1 4H7V7h3v3zm1 3H6v5h5v-5zm-1 4H7v-3h3v3z"/></svg>'), + 'show_ui' => true, + 'show_in_admin_bar' => false, + 'public' => false, + 'rewrite' => false, + 'query_var' => false, + 'capability_type' => 'page', + 'show_in_rest' => true, + 'rest_controller_class' => 'Grunion_Contact_Form_Endpoint', + 'capabilities' => array( + 'create_posts' => false, + 'publish_posts' => 'publish_pages', + 'edit_posts' => 'edit_pages', + 'edit_others_posts' => 'edit_others_pages', + 'delete_posts' => 'delete_pages', + 'delete_others_posts' => 'delete_others_pages', + 'read_private_posts' => 'read_private_pages', + 'edit_post' => 'edit_page', + 'delete_post' => 'delete_page', + 'read_post' => 'read_page', + ), + 'map_meta_cap' => true, + ) + ); + + // Add to REST API post type whitelist + add_filter( 'rest_api_allowed_post_types', array( $this, 'allow_feedback_rest_api_type' ) ); + + // Add "spam" as a post status + register_post_status( + 'spam', array( + 'label' => 'Spam', + 'public' => false, + 'exclude_from_search' => true, + 'show_in_admin_all_list' => false, + 'label_count' => _n_noop( 'Spam <span class="count">(%s)</span>', 'Spam <span class="count">(%s)</span>', 'jetpack' ), + 'protected' => true, + '_builtin' => false, + ) + ); + + // POST handler + if ( + isset( $_SERVER['REQUEST_METHOD'] ) && 'POST' == strtoupper( $_SERVER['REQUEST_METHOD'] ) + && + isset( $_POST['action'] ) && 'grunion-contact-form' == $_POST['action'] + && + isset( $_POST['contact-form-id'] ) + ) { + add_action( 'template_redirect', array( $this, 'process_form_submission' ) ); + } + + /* + Can be dequeued by placing the following in wp-content/themes/yourtheme/functions.php + * + * function remove_grunion_style() { + * wp_deregister_style('grunion.css'); + * } + * add_action('wp_print_styles', 'remove_grunion_style'); + */ + wp_register_style( 'grunion.css', GRUNION_PLUGIN_URL . 'css/grunion.css', array(), JETPACK__VERSION ); + wp_style_add_data( 'grunion.css', 'rtl', 'replace' ); + + self::register_contact_form_blocks(); + } + + private static function register_contact_form_blocks() { + jetpack_register_block( 'jetpack/contact-form', array( + 'render_callback' => array( __CLASS__, 'gutenblock_render_form' ), + ) ); + + // Field render methods. + jetpack_register_block( 'jetpack/field-text', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_text' ), + ) ); + jetpack_register_block( 'jetpack/field-name', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_name' ), + ) ); + jetpack_register_block( 'jetpack/field-email', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_email' ), + ) ); + jetpack_register_block( 'jetpack/field-url', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_url' ), + ) ); + jetpack_register_block( 'jetpack/field-date', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_date' ), + ) ); + jetpack_register_block( 'jetpack/field-telephone', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_telephone' ), + ) ); + jetpack_register_block( 'jetpack/field-textarea', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_textarea' ), + ) ); + jetpack_register_block( 'jetpack/field-checkbox', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_checkbox' ), + ) ); + jetpack_register_block( 'jetpack/field-checkbox-multiple', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_checkbox_multiple' ), + ) ); + jetpack_register_block( 'jetpack/field-radio', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_radio' ), + ) ); + jetpack_register_block( 'jetpack/field-select', array( + 'parent' => array( 'jetpack/contact-form' ), + 'render_callback' => array( __CLASS__, 'gutenblock_render_field_select' ), + ) ); + } + + public static function gutenblock_render_form( $atts, $content ) { + return Grunion_Contact_Form::parse( $atts, do_blocks( $content ) ); + } + + public static function block_attributes_to_shortcode_attributes( $atts, $type ) { + $atts['type'] = $type; + if ( isset( $atts['className'] ) ) { + $atts['class'] = $atts['className']; + unset( $atts['className'] ); + } + + if ( isset( $atts['defaultValue'] ) ) { + $atts['default'] = $atts['defaultValue']; + unset( $atts['defaultValue'] ); + } + + return $atts; + } + + public static function gutenblock_render_field_text( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'text' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_name( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'name' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_email( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'email' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_url( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'url' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_date( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'date' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_telephone( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'telephone' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_textarea( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'textarea' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_checkbox( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'checkbox' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_checkbox_multiple( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'checkbox-multiple' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_radio( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'radio' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + public static function gutenblock_render_field_select( $atts, $content ) { + $atts = self::block_attributes_to_shortcode_attributes( $atts, 'select' ); + return Grunion_Contact_Form::parse_contact_field( $atts, $content ); + } + + /** + * Add the 'Export' menu item as a submenu of Feedback. + */ + public function admin_menu() { + add_submenu_page( + 'edit.php?post_type=feedback', + __( 'Export feedback as CSV', 'jetpack' ), + __( 'Export CSV', 'jetpack' ), + 'export', + 'feedback-export', + array( $this, 'export_form' ) + ); + } + + /** + * Add to REST API post type whitelist + */ + function allow_feedback_rest_api_type( $post_types ) { + $post_types[] = 'feedback'; + return $post_types; + } + + /** + * Display the count of new feedback entries received. It's reset when user visits the Feedback screen. + * + * @since 4.1.0 + * + * @param object $screen Information about the current screen. + */ + function unread_count( $screen ) { + if ( isset( $screen->post_type ) && 'feedback' == $screen->post_type ) { + update_option( 'feedback_unread_count', 0 ); + } else { + global $menu; + if ( isset( $menu ) && is_array( $menu ) && ! empty( $menu ) ) { + foreach ( $menu as $index => $menu_item ) { + if ( 'edit.php?post_type=feedback' == $menu_item[2] ) { + $unread = get_option( 'feedback_unread_count', 0 ); + if ( $unread > 0 ) { + $unread_count = current_user_can( 'publish_pages' ) ? " <span class='feedback-unread count-{$unread} awaiting-mod'><span class='feedback-unread-count'>" . number_format_i18n( $unread ) . '</span></span>' : ''; + $menu[ $index ][0] .= $unread_count; + } + break; + } + } + } + } + } + + /** + * Handles all contact-form POST submissions + * + * Conditionally attached to `template_redirect` + */ + function process_form_submission() { + // Add a filter to replace tokens in the subject field with sanitized field values + add_filter( 'contact_form_subject', array( $this, 'replace_tokens_with_input' ), 10, 2 ); + + $id = stripslashes( $_POST['contact-form-id'] ); + $hash = isset( $_POST['contact-form-hash'] ) ? $_POST['contact-form-hash'] : null; + $hash = preg_replace( '/[^\da-f]/i', '', $hash ); + + if ( is_user_logged_in() ) { + check_admin_referer( "contact-form_{$id}" ); + } + + $is_widget = 0 === strpos( $id, 'widget-' ); + + $form = false; + + if ( $is_widget ) { + // It's a form embedded in a text widget + $this->current_widget_id = substr( $id, 7 ); // remove "widget-" + $widget_type = implode( '-', array_slice( explode( '-', $this->current_widget_id ), 0, -1 ) ); // Remove trailing -# + + // Is the widget active? + $sidebar = is_active_widget( false, $this->current_widget_id, $widget_type ); + + // This is lame - no core API for getting a widget by ID + $widget = isset( $GLOBALS['wp_registered_widgets'][ $this->current_widget_id ] ) ? $GLOBALS['wp_registered_widgets'][ $this->current_widget_id ] : false; + + if ( $sidebar && $widget && isset( $widget['callback'] ) ) { + // prevent PHP notices by populating widget args + $widget_args = array( + 'before_widget' => '', + 'after_widget' => '', + 'before_title' => '', + 'after_title' => '', + ); + // This is lamer - no API for outputting a given widget by ID + ob_start(); + // Process the widget to populate Grunion_Contact_Form::$last + call_user_func( $widget['callback'], $widget_args, $widget['params'][0] ); + ob_end_clean(); + } + } else { + // It's a form embedded in a post + $post = get_post( $id ); + + // Process the content to populate Grunion_Contact_Form::$last + /** This filter is already documented in core. wp-includes/post-template.php */ + apply_filters( 'the_content', $post->post_content ); + } + + $form = isset( Grunion_Contact_Form::$forms[ $hash ] ) ? Grunion_Contact_Form::$forms[ $hash ] : null; + + // No form may mean user is using do_shortcode, grab the form using the stored post meta + if ( ! $form ) { + + // Get shortcode from post meta + $shortcode = get_post_meta( $_POST['contact-form-id'], "_g_feedback_shortcode_{$hash}", true ); + + // Format it + if ( $shortcode != '' ) { + + // Get attributes from post meta. + $parameters = ''; + $attributes = get_post_meta( $_POST['contact-form-id'], "_g_feedback_shortcode_atts_{$hash}", true ); + if ( ! empty( $attributes ) && is_array( $attributes ) ) { + foreach ( array_filter( $attributes ) as $param => $value ) { + $parameters .= " $param=\"$value\""; + } + } + + $shortcode = '[contact-form' . $parameters . ']' . $shortcode . '[/contact-form]'; + do_shortcode( $shortcode ); + + // Recreate form + $form = Grunion_Contact_Form::$last; + } + + if ( ! $form ) { + return false; + } + } + + if ( is_wp_error( $form->errors ) && $form->errors->get_error_codes() ) { + return $form->errors; + } + + // Process the form + return $form->process_submission(); + } + + function ajax_request() { + $submission_result = self::process_form_submission(); + + if ( ! $submission_result ) { + header( 'HTTP/1.1 500 Server Error', 500, true ); + echo '<div class="form-error"><ul class="form-errors"><li class="form-error-message">'; + esc_html_e( 'An error occurred. Please try again later.', 'jetpack' ); + echo '</li></ul></div>'; + } elseif ( is_wp_error( $submission_result ) ) { + header( 'HTTP/1.1 400 Bad Request', 403, true ); + echo '<div class="form-error"><ul class="form-errors"><li class="form-error-message">'; + echo esc_html( $submission_result->get_error_message() ); + echo '</li></ul></div>'; + } else { + echo '<h3>' . esc_html__( 'Message Sent', 'jetpack' ) . '</h3>' . $submission_result; + } + + die; + } + + /** + * Ensure the post author is always zero for contact-form feedbacks + * Attached to `wp_insert_post_data` + * + * @see Grunion_Contact_Form::process_submission() + * + * @param array $data the data to insert + * @param array $postarr the data sent to wp_insert_post() + * @return array The filtered $data to insert + */ + function insert_feedback_filter( $data, $postarr ) { + if ( $data['post_type'] == 'feedback' && $postarr['post_type'] == 'feedback' ) { + $data['post_author'] = 0; + } + + return $data; + } + /* + * Adds our contact-form shortcode + * The "child" contact-field shortcode is enabled as needed by the contact-form shortcode handler + */ + function add_shortcode() { + add_shortcode( 'contact-form', array( 'Grunion_Contact_Form', 'parse' ) ); + add_shortcode( 'contact-field', array( 'Grunion_Contact_Form', 'parse_contact_field' ) ); + } + + static function tokenize_label( $label ) { + return '{' . trim( preg_replace( '#^\d+_#', '', $label ) ) . '}'; + } + + static function sanitize_value( $value ) { + return preg_replace( '=((<CR>|<LF>|0x0A/%0A|0x0D/%0D|\\n|\\r)\S).*=i', null, $value ); + } + + /** + * Replaces tokens like {city} or {City} (case insensitive) with the value + * of an input field of that name + * + * @param string $subject + * @param array $field_values Array with field label => field value associations + * + * @return string The filtered $subject with the tokens replaced + */ + function replace_tokens_with_input( $subject, $field_values ) { + // Wrap labels into tokens (inside {}) + $wrapped_labels = array_map( array( 'Grunion_Contact_Form_Plugin', 'tokenize_label' ), array_keys( $field_values ) ); + // Sanitize all values + $sanitized_values = array_map( array( 'Grunion_Contact_Form_Plugin', 'sanitize_value' ), array_values( $field_values ) ); + + foreach ( $sanitized_values as $k => $sanitized_value ) { + if ( is_array( $sanitized_value ) ) { + $sanitized_values[ $k ] = implode( ', ', $sanitized_value ); + } + } + + // Search for all valid tokens (based on existing fields) and replace with the field's value + $subject = str_ireplace( $wrapped_labels, $sanitized_values, $subject ); + return $subject; + } + + /** + * Tracks the widget currently being processed. + * Attached to `dynamic_sidebar` + * + * @see $current_widget_id + * + * @param array $widget The widget data + */ + function track_current_widget( $widget ) { + $this->current_widget_id = $widget['id']; + } + + /** + * Adds a "widget" attribute to every contact-form embedded in a text widget. + * Used to tell the difference between post-embedded contact-forms and widget-embedded contact-forms + * Attached to `widget_text` + * + * @param string $text The widget text + * @return string The filtered widget text + */ + function widget_atts( $text ) { + Grunion_Contact_Form::style( true ); + + return preg_replace( '/\[contact-form([^a-zA-Z_-])/', '[contact-form widget="' . $this->current_widget_id . '"\\1', $text ); + } + + /** + * For sites where text widgets are not processed for shortcodes, we add this hack to process just our shortcode + * Attached to `widget_text` + * + * @param string $text The widget text + * @return string The contact-form filtered widget text + */ + function widget_shortcode_hack( $text ) { + if ( ! preg_match( '/\[contact-form([^a-zA-Z_-])/', $text ) ) { + return $text; + } + + $old = $GLOBALS['shortcode_tags']; + remove_all_shortcodes(); + Grunion_Contact_Form_Plugin::$using_contact_form_field = true; + $this->add_shortcode(); + + $text = do_shortcode( $text ); + + Grunion_Contact_Form_Plugin::$using_contact_form_field = false; + $GLOBALS['shortcode_tags'] = $old; + + return $text; + } + + /** + * Check if a submission matches the Comment Blacklist. + * The Comment Blacklist is a means to moderate discussion, and contact + * forms are 1:1 discussion forums, ripe for abuse by users who are being + * removed from the public discussion. + * Attached to `jetpack_contact_form_is_spam` + * + * @param bool $is_spam + * @param array $form + * @return bool TRUE => spam, FALSE => not spam + */ + function is_spam_blacklist( $is_spam, $form = array() ) { + if ( $is_spam ) { + return $is_spam; + } + + if ( wp_blacklist_check( $form['comment_author'], $form['comment_author_email'], $form['comment_author_url'], $form['comment_content'], $form['user_ip'], $form['user_agent'] ) ) { + return true; + } + + return false; + } + + /** + * Populate an array with all values necessary to submit a NEW contact-form feedback to Akismet. + * Note that this includes the current user_ip etc, so this should only be called when accepting a new item via $_POST + * + * @param array $form Contact form feedback array + * @return array feedback array with additional data ready for submission to Akismet + */ + function prepare_for_akismet( $form ) { + $form['comment_type'] = 'contact_form'; + $form['user_ip'] = $_SERVER['REMOTE_ADDR']; + $form['user_agent'] = $_SERVER['HTTP_USER_AGENT']; + $form['referrer'] = $_SERVER['HTTP_REFERER']; + $form['blog'] = get_option( 'home' ); + + foreach ( $_SERVER as $key => $value ) { + if ( ! is_string( $value ) ) { + continue; + } + if ( in_array( $key, array( 'HTTP_COOKIE', 'HTTP_COOKIE2', 'HTTP_USER_AGENT', 'HTTP_REFERER' ) ) ) { + // We don't care about cookies, and the UA and Referrer were caught above. + continue; + } elseif ( in_array( $key, array( 'REMOTE_ADDR', 'REQUEST_URI', 'DOCUMENT_URI' ) ) ) { + // All three of these are relevant indicators and should be passed along. + $form[ $key ] = $value; + } elseif ( wp_startswith( $key, 'HTTP_' ) ) { + // Any other HTTP header indicators. + // `wp_startswith()` is a wpcom helper function and is included in Jetpack via `functions.compat.php` + $form[ $key ] = $value; + } + } + + return $form; + } + + /** + * Submit contact-form data to Akismet to check for spam. + * If you're accepting a new item via $_POST, run it Grunion_Contact_Form_Plugin::prepare_for_akismet() first + * Attached to `jetpack_contact_form_is_spam` + * + * @param bool $is_spam + * @param array $form + * @return bool|WP_Error TRUE => spam, FALSE => not spam, WP_Error => stop processing entirely + */ + function is_spam_akismet( $is_spam, $form = array() ) { + global $akismet_api_host, $akismet_api_port; + + // The signature of this function changed from accepting just $form. + // If something only sends an array, assume it's still using the old + // signature and work around it. + if ( empty( $form ) && is_array( $is_spam ) ) { + $form = $is_spam; + $is_spam = false; + } + + // If a previous filter has alrady marked this as spam, trust that and move on. + if ( $is_spam ) { + return $is_spam; + } + + if ( ! function_exists( 'akismet_http_post' ) && ! defined( 'AKISMET_VERSION' ) ) { + return false; + } + + $query_string = http_build_query( $form ); + + if ( method_exists( 'Akismet', 'http_post' ) ) { + $response = Akismet::http_post( $query_string, 'comment-check' ); + } else { + $response = akismet_http_post( $query_string, $akismet_api_host, '/1.1/comment-check', $akismet_api_port ); + } + + $result = false; + + if ( isset( $response[0]['x-akismet-pro-tip'] ) && 'discard' === trim( $response[0]['x-akismet-pro-tip'] ) && get_option( 'akismet_strictness' ) === '1' ) { + $result = new WP_Error( 'feedback-discarded', __( 'Feedback discarded.', 'jetpack' ) ); + } elseif ( isset( $response[1] ) && 'true' == trim( $response[1] ) ) { // 'true' is spam + $result = true; + } + + /** + * Filter the results returned by Akismet for each submitted contact form. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param WP_Error|bool $result Is the submitted feedback spam. + * @param array|bool $form Submitted feedback. + */ + return apply_filters( 'contact_form_is_spam_akismet', $result, $form ); + } + + /** + * Submit a feedback as either spam or ham + * + * @param string $as Either 'spam' or 'ham'. + * @param array $form the contact-form data + */ + function akismet_submit( $as, $form ) { + global $akismet_api_host, $akismet_api_port; + + if ( ! in_array( $as, array( 'ham', 'spam' ) ) ) { + return false; + } + + $query_string = ''; + if ( is_array( $form ) ) { + $query_string = http_build_query( $form ); + } + if ( method_exists( 'Akismet', 'http_post' ) ) { + $response = Akismet::http_post( $query_string, "submit-{$as}" ); + } else { + $response = akismet_http_post( $query_string, $akismet_api_host, "/1.1/submit-{$as}", $akismet_api_port ); + } + + return trim( $response[1] ); + } + + /** + * Prints the menu + */ + function export_form() { + $current_screen = get_current_screen(); + if ( ! in_array( $current_screen->id, array( 'edit-feedback', 'feedback_page_feedback-export' ) ) ) { + return; + } + + if ( ! current_user_can( 'export' ) ) { + return; + } + + // if there aren't any feedbacks, bail out + if ( ! (int) wp_count_posts( 'feedback' )->publish ) { + return; + } + ?> + + <div id="feedback-export" style="display:none"> + <h2><?php _e( 'Export feedback as CSV', 'jetpack' ); ?></h2> + <div class="clear"></div> + <form action="<?php echo admin_url( 'admin-post.php' ); ?>" method="post" class="form"> + <?php wp_nonce_field( 'feedback_export', 'feedback_export_nonce' ); ?> + + <input name="action" value="feedback_export" type="hidden"> + <label for="post"><?php _e( 'Select feedback to download', 'jetpack' ); ?></label> + <select name="post"> + <option value="all"><?php esc_html_e( 'All posts', 'jetpack' ); ?></option> + <?php echo $this->get_feedbacks_as_options(); ?> + </select> + + <br><br> + <input type="submit" name="submit" id="submit" class="button button-primary" value="<?php esc_html_e( 'Download', 'jetpack' ); ?>"> + </form> + </div> + + <?php + // There aren't any usable actions in core to output the "export feedback" form in the correct place, + // so this inline JS moves it from the top of the page to the bottom. + ?> + <script type='text/javascript'> + var menu = document.getElementById( 'feedback-export' ), + wrapper = document.getElementsByClassName( 'wrap' )[0]; + <?php if ( 'edit-feedback' === $current_screen->id ) : ?> + wrapper.appendChild(menu); + <?php endif; ?> + menu.style.display = 'block'; + </script> + <?php + } + + /** + * Fetch post content for a post and extract just the comment. + * + * @param int $post_id The post id to fetch the content for. + * + * @return string Trimmed post comment. + * + * @codeCoverageIgnore + */ + public function get_post_content_for_csv_export( $post_id ) { + $post_content = get_post_field( 'post_content', $post_id ); + $content = explode( '<!--more-->', $post_content ); + + return trim( $content[0] ); + } + + /** + * Get `_feedback_extra_fields` field from post meta data. + * + * @param int $post_id Id of the post to fetch meta data for. + * + * @return mixed + * + * @codeCoverageIgnore - No need to be covered. + */ + public function get_post_meta_for_csv_export( $post_id ) { + return get_post_meta( $post_id, '_feedback_extra_fields', true ); + } + + /** + * Get parsed feedback post fields. + * + * @param int $post_id Id of the post to fetch parsed contents for. + * + * @return array + * + * @codeCoverageIgnore - No need to be covered. + */ + public function get_parsed_field_contents_of_post( $post_id ) { + return self::parse_fields_from_content( $post_id ); + } + + /** + * Properly maps fields that are missing from the post meta data + * to names, that are similar to those of the post meta. + * + * @param array $parsed_post_content Parsed post content + * + * @see parse_fields_from_content for how the input data is generated. + * + * @return array Mapped fields. + */ + public function map_parsed_field_contents_of_post_to_field_names( $parsed_post_content ) { + + $mapped_fields = array(); + + $field_mapping = array( + '_feedback_subject' => __( 'Contact Form', 'jetpack' ), + '_feedback_author' => '1_Name', + '_feedback_author_email' => '2_Email', + '_feedback_author_url' => '3_Website', + '_feedback_main_comment' => '4_Comment', + ); + + foreach ( $field_mapping as $parsed_field_name => $field_name ) { + if ( + isset( $parsed_post_content[ $parsed_field_name ] ) + && ! empty( $parsed_post_content[ $parsed_field_name ] ) + ) { + $mapped_fields[ $field_name ] = $parsed_post_content[ $parsed_field_name ]; + } + } + + return $mapped_fields; + } + + /** + * Registers the personal data exporter. + * + * @since 6.1.1 + * + * @param array $exporters An array of personal data exporters. + * + * @return array $exporters An array of personal data exporters. + */ + public function register_personal_data_exporter( $exporters ) { + $exporters['jetpack-feedback'] = array( + 'exporter_friendly_name' => __( 'Feedback', 'jetpack' ), + 'callback' => array( $this, 'personal_data_exporter' ), + ); + + return $exporters; + } + + /** + * Registers the personal data eraser. + * + * @since 6.1.1 + * + * @param array $erasers An array of personal data erasers. + * + * @return array $erasers An array of personal data erasers. + */ + public function register_personal_data_eraser( $erasers ) { + $erasers['jetpack-feedback'] = array( + 'eraser_friendly_name' => __( 'Feedback', 'jetpack' ), + 'callback' => array( $this, 'personal_data_eraser' ), + ); + + return $erasers; + } + + /** + * Exports personal data. + * + * @since 6.1.1 + * + * @param string $email Email address. + * @param int $page Page to export. + * + * @return array $return Associative array with keys expected by core. + */ + public function personal_data_exporter( $email, $page = 1 ) { + return $this->_internal_personal_data_exporter( $email, $page ); + } + + /** + * Internal method for exporting personal data. + * + * Allows us to have a different signature than core expects + * while protecting against future core API changes. + * + * @internal + * @since 6.5 + * + * @param string $email Email address. + * @param int $page Page to export. + * @param int $per_page Number of feedbacks to process per page. Internal use only (testing) + * + * @return array Associative array with keys expected by core. + */ + public function _internal_personal_data_exporter( $email, $page = 1, $per_page = 250 ) { + $export_data = array(); + $post_ids = $this->personal_data_post_ids_by_email( $email, $per_page, $page ); + + foreach ( $post_ids as $post_id ) { + $post_fields = $this->get_parsed_field_contents_of_post( $post_id ); + + if ( ! is_array( $post_fields ) || empty( $post_fields['_feedback_subject'] ) ) { + continue; // Corrupt data. + } + + $post_fields['_feedback_main_comment'] = $this->get_post_content_for_csv_export( $post_id ); + $post_fields = $this->map_parsed_field_contents_of_post_to_field_names( $post_fields ); + + if ( ! is_array( $post_fields ) || empty( $post_fields ) ) { + continue; // No fields to export. + } + + $post_meta = $this->get_post_meta_for_csv_export( $post_id ); + $post_meta = is_array( $post_meta ) ? $post_meta : array(); + + $post_export_data = array(); + $post_data = array_merge( $post_fields, $post_meta ); + ksort( $post_data ); + + foreach ( $post_data as $post_data_key => $post_data_value ) { + $post_export_data[] = array( + 'name' => preg_replace( '/^[0-9]+_/', '', $post_data_key ), + 'value' => $post_data_value, + ); + } + + $export_data[] = array( + 'group_id' => 'feedback', + 'group_label' => __( 'Feedback', 'jetpack' ), + 'item_id' => 'feedback-' . $post_id, + 'data' => $post_export_data, + ); + } + + return array( + 'data' => $export_data, + 'done' => count( $post_ids ) < $per_page, + ); + } + + /** + * Erases personal data. + * + * @since 6.1.1 + * + * @param string $email Email address. + * @param int $page Page to erase. + * + * @return array Associative array with keys expected by core. + */ + public function personal_data_eraser( $email, $page = 1 ) { + return $this->_internal_personal_data_eraser( $email, $page ); + } + + /** + * Internal method for erasing personal data. + * + * Allows us to have a different signature than core expects + * while protecting against future core API changes. + * + * @internal + * @since 6.5 + * + * @param string $email Email address. + * @param int $page Page to erase. + * @param int $per_page Number of feedbacks to process per page. Internal use only (testing) + * + * @return array Associative array with keys expected by core. + */ + public function _internal_personal_data_eraser( $email, $page = 1, $per_page = 250 ) { + $removed = false; + $retained = false; + $messages = array(); + $option_name = sprintf( '_jetpack_pde_feedback_%s', md5( $email ) ); + $last_post_id = 1 === $page ? 0 : get_option( $option_name, 0 ); + $post_ids = $this->personal_data_post_ids_by_email( $email, $per_page, $page, $last_post_id ); + + foreach ( $post_ids as $post_id ) { + /** + * Filters whether to erase a particular Feedback post. + * + * @since 6.3.0 + * + * @param bool|string $prevention_message Whether to apply erase the Feedback post (bool). + * Custom prevention message (string). Default true. + * @param int $post_id Feedback post ID. + */ + $prevention_message = apply_filters( 'grunion_contact_form_delete_feedback_post', true, $post_id ); + + if ( true !== $prevention_message ) { + if ( $prevention_message && is_string( $prevention_message ) ) { + $messages[] = esc_html( $prevention_message ); + } else { + $messages[] = sprintf( + // translators: %d: Post ID. + __( 'Feedback ID %d could not be removed at this time.', 'jetpack' ), + $post_id + ); + } + + $retained = true; + + continue; + } + + if ( wp_delete_post( $post_id, true ) ) { + $removed = true; + } else { + $retained = true; + $messages[] = sprintf( + // translators: %d: Post ID. + __( 'Feedback ID %d could not be removed at this time.', 'jetpack' ), + $post_id + ); + } + } + + $done = count( $post_ids ) < $per_page; + + if ( $done ) { + delete_option( $option_name ); + } else { + update_option( $option_name, (int) $post_id ); + } + + return array( + 'items_removed' => $removed, + 'items_retained' => $retained, + 'messages' => $messages, + 'done' => $done, + ); + } + + /** + * Queries personal data by email address. + * + * @since 6.1.1 + * + * @param string $email Email address. + * @param int $per_page Post IDs per page. Default is `250`. + * @param int $page Page to query. Default is `1`. + * @param int $last_post_id Page to query. Default is `0`. If non-zero, used instead of $page. + * + * @return array An array of post IDs. + */ + public function personal_data_post_ids_by_email( $email, $per_page = 250, $page = 1, $last_post_id = 0 ) { + add_filter( 'posts_search', array( $this, 'personal_data_search_filter' ) ); + + $this->pde_last_post_id_erased = $last_post_id; + $this->pde_email_address = $email; + + $post_ids = get_posts( + array( + 'post_type' => 'feedback', + 'post_status' => 'publish', + // This search parameter gets overwritten in ->personal_data_search_filter() + 's' => '..PDE..AUTHOR EMAIL:..PDE..', + 'sentence' => true, + 'order' => 'ASC', + 'orderby' => 'ID', + 'fields' => 'ids', + 'posts_per_page' => $per_page, + 'paged' => $last_post_id ? 1 : $page, + 'suppress_filters' => false, + ) + ); + + $this->pde_last_post_id_erased = 0; + $this->pde_email_address = ''; + + remove_filter( 'posts_search', array( $this, 'personal_data_search_filter' ) ); + + return $post_ids; + } + + /** + * Filters searches by email address. + * + * @since 6.1.1 + * + * @param string $search SQL where clause. + * + * @return array Filtered SQL where clause. + */ + public function personal_data_search_filter( $search ) { + global $wpdb; + + /* + * Limits search to `post_content` only, and we only match the + * author's email address whenever it's on a line by itself. + */ + if ( $this->pde_email_address && false !== strpos( $search, '..PDE..AUTHOR EMAIL:..PDE..' ) ) { + $search = $wpdb->prepare( + " AND ( + {$wpdb->posts}.post_content LIKE %s + OR {$wpdb->posts}.post_content LIKE %s + )", + // `chr( 10 )` = `\n`, `chr( 13 )` = `\r` + '%' . $wpdb->esc_like( chr( 10 ) . 'AUTHOR EMAIL: ' . $this->pde_email_address . chr( 10 ) ) . '%', + '%' . $wpdb->esc_like( chr( 13 ) . 'AUTHOR EMAIL: ' . $this->pde_email_address . chr( 13 ) ) . '%' + ); + + if ( $this->pde_last_post_id_erased ) { + $search .= $wpdb->prepare( " AND {$wpdb->posts}.ID > %d", $this->pde_last_post_id_erased ); + } + } + + return $search; + } + + /** + * Prepares feedback post data for CSV export. + * + * @param array $post_ids Post IDs to fetch the data for. These need to be Feedback posts. + * + * @return array + */ + public function get_export_data_for_posts( $post_ids ) { + + $posts_data = array(); + $field_names = array(); + $result = array(); + + /** + * Fetch posts and get the possible field names for later use + */ + foreach ( $post_ids as $post_id ) { + + /** + * Fetch post main data, because we need the subject and author data for the feedback form. + */ + $post_real_data = $this->get_parsed_field_contents_of_post( $post_id ); + + /** + * If `$post_real_data` is not an array or there is no `_feedback_subject` set, + * then something must be wrong with the feedback post. Skip it. + */ + if ( ! is_array( $post_real_data ) || ! isset( $post_real_data['_feedback_subject'] ) ) { + continue; + } + + /** + * Fetch main post comment. This is from the default textarea fields. + * If it is non-empty, then we add it to data, otherwise skip it. + */ + $post_comment_content = $this->get_post_content_for_csv_export( $post_id ); + if ( ! empty( $post_comment_content ) ) { + $post_real_data['_feedback_main_comment'] = $post_comment_content; + } + + /** + * Map parsed fields to proper field names + */ + $mapped_fields = $this->map_parsed_field_contents_of_post_to_field_names( $post_real_data ); + + /** + * Fetch post meta data. + */ + $post_meta_data = $this->get_post_meta_for_csv_export( $post_id ); + + /** + * If `$post_meta_data` is not an array or if it is empty, then there is no + * extra feedback to work with. Create an empty array. + */ + if ( ! is_array( $post_meta_data ) || empty( $post_meta_data ) ) { + $post_meta_data = array(); + } + + /** + * Prepend the feedback subject to the list of fields. + */ + $post_meta_data = array_merge( + $mapped_fields, + $post_meta_data + ); + + /** + * Save post metadata for later usage. + */ + $posts_data[ $post_id ] = $post_meta_data; + + /** + * Save field names, so we can use them as header fields later in the CSV. + */ + $field_names = array_merge( $field_names, array_keys( $post_meta_data ) ); + } + + /** + * Make sure the field names are unique, because we don't want duplicate data. + */ + $field_names = array_unique( $field_names ); + + /** + * Sort the field names by the field id number + */ + sort( $field_names, SORT_NUMERIC ); + + /** + * Loop through every post, which is essentially CSV row. + */ + foreach ( $posts_data as $post_id => $single_post_data ) { + + /** + * Go through all the possible fields and check if the field is available + * in the current post. + * + * If it is - add the data as a value. + * If it is not - add an empty string, which is just a placeholder in the CSV. + */ + foreach ( $field_names as $single_field_name ) { + if ( + isset( $single_post_data[ $single_field_name ] ) + && ! empty( $single_post_data[ $single_field_name ] ) + ) { + $result[ $single_field_name ][] = trim( $single_post_data[ $single_field_name ] ); + } else { + $result[ $single_field_name ][] = ''; + } + } + } + + return $result; + } + + /** + * download as a csv a contact form or all of them in a csv file + */ + function download_feedback_as_csv() { + if ( empty( $_POST['feedback_export_nonce'] ) ) { + return; + } + + check_admin_referer( 'feedback_export', 'feedback_export_nonce' ); + + if ( ! current_user_can( 'export' ) ) { + return; + } + + $args = array( + 'posts_per_page' => -1, + 'post_type' => 'feedback', + 'post_status' => 'publish', + 'order' => 'ASC', + 'fields' => 'ids', + 'suppress_filters' => false, + ); + + $filename = date( 'Y-m-d' ) . '-feedback-export.csv'; + + // Check if we want to download all the feedbacks or just a certain contact form + if ( ! empty( $_POST['post'] ) && $_POST['post'] !== 'all' ) { + $args['post_parent'] = (int) $_POST['post']; + $filename = date( 'Y-m-d' ) . '-' . str_replace( ' ', '-', get_the_title( (int) $_POST['post'] ) ) . '.csv'; + } + + $feedbacks = get_posts( $args ); + + if ( empty( $feedbacks ) ) { + return; + } + + $filename = sanitize_file_name( $filename ); + + /** + * Prepare data for export. + */ + $data = $this->get_export_data_for_posts( $feedbacks ); + + /** + * If `$data` is empty, there's nothing we can do below. + */ + if ( ! is_array( $data ) || empty( $data ) ) { + return; + } + + /** + * Extract field names from `$data` for later use. + */ + $fields = array_keys( $data ); + + /** + * Count how many rows will be exported. + */ + $row_count = count( reset( $data ) ); + + // Forces the download of the CSV instead of echoing + header( 'Content-Disposition: attachment; filename=' . $filename ); + header( 'Pragma: no-cache' ); + header( 'Expires: 0' ); + header( 'Content-Type: text/csv; charset=utf-8' ); + + $output = fopen( 'php://output', 'w' ); + + /** + * Print CSV headers + */ + fputcsv( $output, $fields ); + + /** + * Print rows to the output. + */ + for ( $i = 0; $i < $row_count; $i ++ ) { + + $current_row = array(); + + /** + * Put all the fields in `$current_row` array. + */ + foreach ( $fields as $single_field_name ) { + $current_row[] = $this->esc_csv( $data[ $single_field_name ][ $i ] ); + } + + /** + * Output the complete CSV row + */ + fputcsv( $output, $current_row ); + } + + fclose( $output ); + } + + /** + * Escape a string to be used in a CSV context + * + * Malicious input can inject formulas into CSV files, opening up the possibility for phishing attacks and + * disclosure of sensitive information. + * + * Additionally, Excel exposes the ability to launch arbitrary commands through the DDE protocol. + * + * @see http://www.contextis.com/resources/blog/comma-separated-vulnerabilities/ + * + * @param string $field + * + * @return string + */ + public function esc_csv( $field ) { + $active_content_triggers = array( '=', '+', '-', '@' ); + + if ( in_array( mb_substr( $field, 0, 1 ), $active_content_triggers, true ) ) { + $field = "'" . $field; + } + + return $field; + } + + /** + * Returns a string of HTML <option> items from an array of posts + * + * @return string a string of HTML <option> items + */ + protected function get_feedbacks_as_options() { + $options = ''; + + // Get the feedbacks' parents' post IDs + $feedbacks = get_posts( + array( + 'fields' => 'id=>parent', + 'posts_per_page' => 100000, + 'post_type' => 'feedback', + 'post_status' => 'publish', + 'suppress_filters' => false, + ) + ); + $parents = array_unique( array_values( $feedbacks ) ); + + $posts = get_posts( + array( + 'orderby' => 'ID', + 'posts_per_page' => 1000, + 'post_type' => 'any', + 'post__in' => array_values( $parents ), + 'suppress_filters' => false, + ) + ); + + // creates the string of <option> elements + foreach ( $posts as $post ) { + $options .= sprintf( '<option value="%s">%s</option>', esc_attr( $post->ID ), esc_html( $post->post_title ) ); + } + + return $options; + } + + /** + * Get the names of all the form's fields + * + * @param array|int $posts the post we want the fields of + * + * @return array the array of fields + * + * @deprecated As this is no longer necessary as of the CSV export rewrite. - 2015-12-29 + */ + protected function get_field_names( $posts ) { + $posts = (array) $posts; + $all_fields = array(); + + foreach ( $posts as $post ) { + $fields = self::parse_fields_from_content( $post ); + + if ( isset( $fields['_feedback_all_fields'] ) ) { + $extra_fields = array_keys( $fields['_feedback_all_fields'] ); + $all_fields = array_merge( $all_fields, $extra_fields ); + } + } + + $all_fields = array_unique( $all_fields ); + return $all_fields; + } + + public static function parse_fields_from_content( $post_id ) { + static $post_fields; + + if ( ! is_array( $post_fields ) ) { + $post_fields = array(); + } + + if ( isset( $post_fields[ $post_id ] ) ) { + return $post_fields[ $post_id ]; + } + + $all_values = array(); + $post_content = get_post_field( 'post_content', $post_id ); + $content = explode( '<!--more-->', $post_content ); + $lines = array(); + + if ( count( $content ) > 1 ) { + $content = str_ireplace( array( '<br />', ')</p>' ), '', $content[1] ); + $one_line = preg_replace( '/\s+/', ' ', $content ); + $one_line = preg_replace( '/.*Array \( (.*)\)/', '$1', $one_line ); + + preg_match_all( '/\[([^\]]+)\] =\>\; ([^\[]+)/', $one_line, $matches ); + + if ( count( $matches ) > 1 ) { + $all_values = array_combine( array_map( 'trim', $matches[1] ), array_map( 'trim', $matches[2] ) ); + } + + $lines = array_filter( explode( "\n", $content ) ); + } + + $var_map = array( + 'AUTHOR' => '_feedback_author', + 'AUTHOR EMAIL' => '_feedback_author_email', + 'AUTHOR URL' => '_feedback_author_url', + 'SUBJECT' => '_feedback_subject', + 'IP' => '_feedback_ip', + ); + + $fields = array(); + + foreach ( $lines as $line ) { + $vars = explode( ': ', $line, 2 ); + if ( ! empty( $vars ) ) { + if ( isset( $var_map[ $vars[0] ] ) ) { + $fields[ $var_map[ $vars[0] ] ] = self::strip_tags( trim( $vars[1] ) ); + } + } + } + + $fields['_feedback_all_fields'] = $all_values; + + $post_fields[ $post_id ] = $fields; + + return $fields; + } + + /** + * Creates a valid csv row from a post id + * + * @param int $post_id The id of the post + * @param array $fields An array containing the names of all the fields of the csv + * @return String The csv row + * + * @deprecated This is no longer needed, as of the CSV export rewrite. + */ + protected static function make_csv_row_from_feedback( $post_id, $fields ) { + $content_fields = self::parse_fields_from_content( $post_id ); + $all_fields = array(); + + if ( isset( $content_fields['_feedback_all_fields'] ) ) { + $all_fields = $content_fields['_feedback_all_fields']; + } + + // Overwrite the parsed content with the content we stored in post_meta in a better format. + $extra_fields = get_post_meta( $post_id, '_feedback_extra_fields', true ); + foreach ( $extra_fields as $extra_field => $extra_value ) { + $all_fields[ $extra_field ] = $extra_value; + } + + // The first element in all of the exports will be the subject + $row_items[] = $content_fields['_feedback_subject']; + + // Loop the fields array in order to fill the $row_items array correctly + foreach ( $fields as $field ) { + if ( $field === __( 'Contact Form', 'jetpack' ) ) { // the first field will ever be the contact form, so we can continue + continue; + } elseif ( array_key_exists( $field, $all_fields ) ) { + $row_items[] = $all_fields[ $field ]; + } else { + $row_items[] = ''; + } + } + + return $row_items; + } + + public static function get_ip_address() { + return isset( $_SERVER['REMOTE_ADDR'] ) ? $_SERVER['REMOTE_ADDR'] : null; + } +} + +/** + * Generic shortcode class. + * Does nothing other than store structured data and output the shortcode as a string + * + * Not very general - specific to Grunion. + */ +class Crunion_Contact_Form_Shortcode { + /** + * @var string the name of the shortcode: [$shortcode_name /] + */ + public $shortcode_name; + + /** + * @var array key => value pairs for the shortcode's attributes: [$shortcode_name key="value" ... /] + */ + public $attributes; + + /** + * @var array key => value pair for attribute defaults + */ + public $defaults = array(); + + /** + * @var null|string Null for selfclosing shortcodes. Hhe inner content of otherwise: [$shortcode_name]$content[/$shortcode_name] + */ + public $content; + + /** + * @var array Associative array of inner "child" shortcodes equivalent to the $content: [$shortcode_name][child 1/][child 2/][/$shortcode_name] + */ + public $fields; + + /** + * @var null|string The HTML of the parsed inner "child" shortcodes". Null for selfclosing shortcodes. + */ + public $body; + + /** + * @param array $attributes An associative array of shortcode attributes. @see shortcode_atts() + * @param null|string $content Null for selfclosing shortcodes. The inner content otherwise. + */ + function __construct( $attributes, $content = null ) { + $this->attributes = $this->unesc_attr( $attributes ); + if ( is_array( $content ) ) { + $string_content = ''; + foreach ( $content as $field ) { + $string_content .= (string) $field; + } + + $this->content = $string_content; + } else { + $this->content = $content; + } + + $this->parse_content( $this->content ); + } + + /** + * Processes the shortcode's inner content for "child" shortcodes + * + * @param string $content The shortcode's inner content: [shortcode]$content[/shortcode] + */ + function parse_content( $content ) { + if ( is_null( $content ) ) { + $this->body = null; + } + + $this->body = do_shortcode( $content ); + } + + /** + * Returns the value of the requested attribute. + * + * @param string $key The attribute to retrieve + * @return mixed + */ + function get_attribute( $key ) { + return isset( $this->attributes[ $key ] ) ? $this->attributes[ $key ] : null; + } + + function esc_attr( $value ) { + if ( is_array( $value ) ) { + return array_map( array( $this, 'esc_attr' ), $value ); + } + + $value = Grunion_Contact_Form_Plugin::strip_tags( $value ); + $value = _wp_specialchars( $value, ENT_QUOTES, false, true ); + + // Shortcode attributes can't contain "]" + $value = str_replace( ']', '', $value ); + $value = str_replace( ',', ',', $value ); // store commas encoded + $value = strtr( + $value, array( + '%' => '%25', + '&' => '%26', + ) + ); + + // shortcode_parse_atts() does stripcslashes() + $value = addslashes( $value ); + return $value; + } + + function unesc_attr( $value ) { + if ( is_array( $value ) ) { + return array_map( array( $this, 'unesc_attr' ), $value ); + } + + // For back-compat with old Grunion encoding + // Also, unencode commas + $value = strtr( + $value, array( + '%26' => '&', + '%25' => '%', + ) + ); + $value = preg_replace( array( '/�*22;/i', '/�*27;/i', '/�*26;/i', '/�*2c;/i' ), array( '"', "'", '&', ',' ), $value ); + $value = htmlspecialchars_decode( $value, ENT_QUOTES ); + $value = Grunion_Contact_Form_Plugin::strip_tags( $value ); + + return $value; + } + + /** + * Generates the shortcode + */ + function __toString() { + $r = "[{$this->shortcode_name} "; + + foreach ( $this->attributes as $key => $value ) { + if ( ! $value ) { + continue; + } + + if ( isset( $this->defaults[ $key ] ) && $this->defaults[ $key ] == $value ) { + continue; + } + + if ( 'id' == $key ) { + continue; + } + + $value = $this->esc_attr( $value ); + + if ( is_array( $value ) ) { + $value = join( ',', $value ); + } + + if ( false === strpos( $value, "'" ) ) { + $value = "'$value'"; + } elseif ( false === strpos( $value, '"' ) ) { + $value = '"' . $value . '"'; + } else { + // Shortcodes can't contain both '"' and "'". Strip one. + $value = str_replace( "'", '', $value ); + $value = "'$value'"; + } + + $r .= "{$key}={$value} "; + } + + $r = rtrim( $r ); + + if ( $this->fields ) { + $r .= ']'; + + foreach ( $this->fields as $field ) { + $r .= (string) $field; + } + + $r .= "[/{$this->shortcode_name}]"; + } else { + $r .= '/]'; + } + + return $r; + } +} + +/** + * Class for the contact-form shortcode. + * Parses shortcode to output the contact form as HTML + * Sends email and stores the contact form response (a.k.a. "feedback") + */ +class Grunion_Contact_Form extends Crunion_Contact_Form_Shortcode { + public $shortcode_name = 'contact-form'; + + /** + * @var WP_Error stores form submission errors + */ + public $errors; + + /** + * @var string The SHA1 hash of the attributes that comprise the form. + */ + public $hash; + + /** + * @var Grunion_Contact_Form The most recent (inclusive) contact-form shortcode processed + */ + static $last; + + /** + * @var Whatever form we are currently looking at. If processed, will become $last + */ + static $current_form; + + /** + * @var array All found forms, indexed by hash. + */ + static $forms = array(); + + /** + * @var bool Whether to print the grunion.css style when processing the contact-form shortcode + */ + static $style = false; + + /** + * @var array When printing the submit button, what tags are allowed + */ + static $allowed_html_tags_for_submit_button = array( 'br' => array() ); + + function __construct( $attributes, $content = null ) { + global $post; + + $this->hash = sha1( json_encode( $attributes ) . $content ); + self::$forms[ $this->hash ] = $this; + + // Set up the default subject and recipient for this form + $default_to = ''; + $default_subject = '[' . get_option( 'blogname' ) . ']'; + + if ( ! isset( $attributes ) || ! is_array( $attributes ) ) { + $attributes = array(); + } + + if ( ! empty( $attributes['widget'] ) && $attributes['widget'] ) { + $default_to .= get_option( 'admin_email' ); + $attributes['id'] = 'widget-' . $attributes['widget']; + $default_subject = sprintf( _x( '%1$s Sidebar', '%1$s = blog name', 'jetpack' ), $default_subject ); + } elseif ( $post ) { + $attributes['id'] = $post->ID; + $default_subject = sprintf( _x( '%1$s %2$s', '%1$s = blog name, %2$s = post title', 'jetpack' ), $default_subject, Grunion_Contact_Form_Plugin::strip_tags( $post->post_title ) ); + $post_author = get_userdata( $post->post_author ); + $default_to .= $post_author->user_email; + } + + // Keep reference to $this for parsing form fields + self::$current_form = $this; + + $this->defaults = array( + 'to' => $default_to, + 'subject' => $default_subject, + 'show_subject' => 'no', // only used in back-compat mode + 'widget' => 0, // Not exposed to the user. Works with Grunion_Contact_Form_Plugin::widget_atts() + 'id' => null, // Not exposed to the user. Set above. + 'submit_button_text' => __( 'Submit', 'jetpack' ), + ); + + $attributes = shortcode_atts( $this->defaults, $attributes, 'contact-form' ); + + // We only enable the contact-field shortcode temporarily while processing the contact-form shortcode + Grunion_Contact_Form_Plugin::$using_contact_form_field = true; + + parent::__construct( $attributes, $content ); + + // There were no fields in the contact form. The form was probably just [contact-form /]. Build a default form. + if ( empty( $this->fields ) ) { + // same as the original Grunion v1 form + $default_form = ' + [contact-field label="' . __( 'Name', 'jetpack' ) . '" type="name" required="true" /] + [contact-field label="' . __( 'Email', 'jetpack' ) . '" type="email" required="true" /] + [contact-field label="' . __( 'Website', 'jetpack' ) . '" type="url" /]'; + + if ( 'yes' == strtolower( $this->get_attribute( 'show_subject' ) ) ) { + $default_form .= ' + [contact-field label="' . __( 'Subject', 'jetpack' ) . '" type="subject" /]'; + } + + $default_form .= ' + [contact-field label="' . __( 'Message', 'jetpack' ) . '" type="textarea" /]'; + + $this->parse_content( $default_form ); + + // Store the shortcode + $this->store_shortcode( $default_form, $attributes, $this->hash ); + } else { + // Store the shortcode + $this->store_shortcode( $content, $attributes, $this->hash ); + } + + // $this->body and $this->fields have been setup. We no longer need the contact-field shortcode. + Grunion_Contact_Form_Plugin::$using_contact_form_field = false; + } + + /** + * Store shortcode content for recall later + * - used to receate shortcode when user uses do_shortcode + * + * @param string $content + * @param array $attributes + * @param string $hash + */ + static function store_shortcode( $content = null, $attributes = null, $hash = null ) { + + if ( $content != null and isset( $attributes['id'] ) ) { + + if ( empty( $hash ) ) { + $hash = sha1( json_encode( $attributes ) . $content ); + } + + $shortcode_meta = get_post_meta( $attributes['id'], "_g_feedback_shortcode_{$hash}", true ); + + if ( $shortcode_meta != '' or $shortcode_meta != $content ) { + update_post_meta( $attributes['id'], "_g_feedback_shortcode_{$hash}", $content ); + + // Save attributes to post_meta for later use. They're not available later in do_shortcode situations. + update_post_meta( $attributes['id'], "_g_feedback_shortcode_atts_{$hash}", $attributes ); + } + } + } + + /** + * Toggle for printing the grunion.css stylesheet + * + * @param bool $style + */ + static function style( $style ) { + $previous_style = self::$style; + self::$style = (bool) $style; + return $previous_style; + } + + /** + * Turn on printing of grunion.css stylesheet + * + * @see ::style() + * @internal + * @param bool $style + */ + static function _style_on() { + return self::style( true ); + } + + /** + * The contact-form shortcode processor + * + * @param array $attributes Key => Value pairs as parsed by shortcode_parse_atts() + * @param string|null $content The shortcode's inner content: [contact-form]$content[/contact-form] + * @return string HTML for the concat form. + */ + static function parse( $attributes, $content ) { + require_once JETPACK__PLUGIN_DIR . '/sync/class.jetpack-sync-settings.php'; + if ( Jetpack_Sync_Settings::is_syncing() ) { + return ''; + } + // Create a new Grunion_Contact_Form object (this class) + $form = new Grunion_Contact_Form( $attributes, $content ); + + $id = $form->get_attribute( 'id' ); + + if ( ! $id ) { // something terrible has happened + return '[contact-form]'; + } + + if ( is_feed() ) { + return '[contact-form]'; + } + + self::$last = $form; + + // Enqueue the grunion.css stylesheet if self::$style allows it + if ( self::$style && ( empty( $_REQUEST['action'] ) || $_REQUEST['action'] != 'grunion_shortcode_to_json' ) ) { + // Enqueue the style here instead of printing it, because if some other plugin has run the_post()+rewind_posts(), + // (like VideoPress does), the style tag gets "printed" the first time and discarded, leaving the contact form unstyled. + // when WordPress does the real loop. + wp_enqueue_style( 'grunion.css' ); + } + + $r = ''; + $r .= "<div id='contact-form-$id'>\n"; + + if ( is_wp_error( $form->errors ) && $form->errors->get_error_codes() ) { + // There are errors. Display them + $r .= "<div class='form-error'>\n<h3>" . __( 'Error!', 'jetpack' ) . "</h3>\n<ul class='form-errors'>\n"; + foreach ( $form->errors->get_error_messages() as $message ) { + $r .= "\t<li class='form-error-message'>" . esc_html( $message ) . "</li>\n"; + } + $r .= "</ul>\n</div>\n\n"; + } + + if ( isset( $_GET['contact-form-id'] ) + && $_GET['contact-form-id'] == self::$last->get_attribute( 'id' ) + && isset( $_GET['contact-form-sent'], $_GET['contact-form-hash'] ) + && hash_equals( $form->hash, $_GET['contact-form-hash'] ) ) { + // The contact form was submitted. Show the success message/results + $feedback_id = (int) $_GET['contact-form-sent']; + + $back_url = remove_query_arg( array( 'contact-form-id', 'contact-form-sent', '_wpnonce' ) ); + + $r_success_message = + '<h3>' . __( 'Message Sent', 'jetpack' ) . + ' (<a href="' . esc_url( $back_url ) . '">' . esc_html__( 'go back', 'jetpack' ) . '</a>)' . + "</h3>\n\n"; + + // Don't show the feedback details unless the nonce matches + if ( $feedback_id && wp_verify_nonce( stripslashes( $_GET['_wpnonce'] ), "contact-form-sent-{$feedback_id}" ) ) { + $r_success_message .= self::success_message( $feedback_id, $form ); + } + + /** + * Filter the message returned after a successful contact form submission. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param string $r_success_message Success message. + */ + $r .= apply_filters( 'grunion_contact_form_success_message', $r_success_message ); + } else { + // Nothing special - show the normal contact form + if ( $form->get_attribute( 'widget' ) ) { + // Submit form to the current URL + $url = remove_query_arg( array( 'contact-form-id', 'contact-form-sent', 'action', '_wpnonce' ) ); + } else { + // Submit form to the post permalink + $url = get_permalink(); + } + + // For SSL/TLS page. See RFC 3986 Section 4.2 + $url = set_url_scheme( $url ); + + // May eventually want to send this to admin-post.php... + /** + * Filter the contact form action URL. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param string $contact_form_id Contact form post URL. + * @param $post $GLOBALS['post'] Post global variable. + * @param int $id Contact Form ID. + */ + $url = apply_filters( 'grunion_contact_form_form_action', "{$url}#contact-form-{$id}", $GLOBALS['post'], $id ); + + $r .= "<form action='" . esc_url( $url ) . "' method='post' class='contact-form commentsblock'>\n"; + $r .= $form->body; + $r .= "\t<p class='contact-submit'>\n"; + + $gutenberg_submit_button_classes = ''; + if ( ! empty( $attributes['submitButtonClasses'] ) ) { + $gutenberg_submit_button_classes = ' ' . $attributes['submitButtonClasses']; + } + + /** + * Filter the contact form submit button class attribute. + * + * @module contact-form + * + * @since 6.6.0 + * + * @param string $class Additional CSS classes for button attribute. + */ + $submit_button_class = apply_filters( 'jetpack_contact_form_submit_button_class', 'pushbutton-wide' . $gutenberg_submit_button_classes ); + + $submit_button_styles = ''; + if ( ! empty( $attributes['customBackgroundButtonColor'] ) ) { + $submit_button_styles .= 'background-color: ' . $attributes['customBackgroundButtonColor'] . '; '; + } + if ( ! empty( $attributes['customTextButtonColor'] ) ) { + $submit_button_styles .= 'color: ' . $attributes['customTextButtonColor'] . ';'; + } + if ( ! empty( $attributes['submitButtonText'] ) ) { + $submit_button_text = $attributes['submitButtonText']; + } else { + $submit_button_text = $form->get_attribute( 'submit_button_text' ); + } + + $r .= "\t\t<button type='submit' class='" . esc_attr( $submit_button_class ) . "'"; + if ( ! empty( $submit_button_styles ) ) { + $r .= " style='" . esc_attr( $submit_button_styles ) . "'"; + } + $r .= ">"; + $r .= wp_kses( + $submit_button_text, + self::$allowed_html_tags_for_submit_button + ) . "</button>"; + + if ( is_user_logged_in() ) { + $r .= "\t\t" . wp_nonce_field( 'contact-form_' . $id, '_wpnonce', true, false ) . "\n"; // nonce and referer + } + $r .= "\t\t<input type='hidden' name='contact-form-id' value='$id' />\n"; + $r .= "\t\t<input type='hidden' name='action' value='grunion-contact-form' />\n"; + $r .= "\t\t<input type='hidden' name='contact-form-hash' value='" . esc_attr( $form->hash ) . "' />\n"; + $r .= "\t</p>\n"; + $r .= "</form>\n"; + } + + $r .= '</div>'; + + return $r; + } + + /** + * Returns a success message to be returned if the form is sent via AJAX. + * + * @param int $feedback_id + * @param object Grunion_Contact_Form $form + * + * @return string $message + */ + static function success_message( $feedback_id, $form ) { + return wp_kses( + '<blockquote class="contact-form-submission">' + . '<p>' . join( self::get_compiled_form( $feedback_id, $form ), '</p><p>' ) . '</p>' + . '</blockquote>', + array( + 'br' => array(), + 'blockquote' => array( 'class' => array() ), + 'p' => array(), + ) + ); + } + + /** + * Returns a compiled form with labels and values in a form of an array + * of lines. + * + * @param int $feedback_id + * @param object Grunion_Contact_Form $form + * + * @return array $lines + */ + static function get_compiled_form( $feedback_id, $form ) { + $feedback = get_post( $feedback_id ); + $field_ids = $form->get_field_ids(); + $content_fields = Grunion_Contact_Form_Plugin::parse_fields_from_content( $feedback_id ); + + // Maps field_ids to post_meta keys + $field_value_map = array( + 'name' => 'author', + 'email' => 'author_email', + 'url' => 'author_url', + 'subject' => 'subject', + 'textarea' => false, // not a post_meta key. This is stored in post_content + ); + + $compiled_form = array(); + + // "Standard" field whitelist + foreach ( $field_value_map as $type => $meta_key ) { + if ( isset( $field_ids[ $type ] ) ) { + $field = $form->fields[ $field_ids[ $type ] ]; + + if ( $meta_key ) { + if ( isset( $content_fields[ "_feedback_{$meta_key}" ] ) ) { + $value = $content_fields[ "_feedback_{$meta_key}" ]; + } + } else { + // The feedback content is stored as the first "half" of post_content + $value = $feedback->post_content; + list( $value ) = explode( '<!--more-->', $value ); + $value = trim( $value ); + } + + $field_index = array_search( $field_ids[ $type ], $field_ids['all'] ); + $compiled_form[ $field_index ] = sprintf( + '<b>%1$s:</b> %2$s<br /><br />', + wp_kses( $field->get_attribute( 'label' ), array() ), + self::escape_and_sanitize_field_value( $value ) + ); + } + } + + // "Non-standard" fields + if ( $field_ids['extra'] ) { + // array indexed by field label (not field id) + $extra_fields = get_post_meta( $feedback_id, '_feedback_extra_fields', true ); + + /** + * Only get data for the compiled form if `$extra_fields` is a valid and non-empty array. + */ + if ( is_array( $extra_fields ) && ! empty( $extra_fields ) ) { + + $extra_field_keys = array_keys( $extra_fields ); + + $i = 0; + foreach ( $field_ids['extra'] as $field_id ) { + $field = $form->fields[ $field_id ]; + $field_index = array_search( $field_id, $field_ids['all'] ); + + $label = $field->get_attribute( 'label' ); + + $compiled_form[ $field_index ] = sprintf( + '<b>%1$s:</b> %2$s<br /><br />', + wp_kses( $label, array() ), + self::escape_and_sanitize_field_value( $extra_fields[ $extra_field_keys[ $i ] ] ) + ); + + $i++; + } + } + } + + // Sorting lines by the field index + ksort( $compiled_form ); + + return $compiled_form; + } + + static function escape_and_sanitize_field_value( $value ) { + $value = str_replace( array( '[' , ']' ) , array( '[' , ']' ) , $value ); + return nl2br( wp_kses( $value, array() ) ); + } + + /** + * Only strip out empty string values and keep all the other values as they are. + * + * @param $single_value + * + * @return bool + */ + static function remove_empty( $single_value ) { + return ( $single_value !== '' ); + } + + /** + * The contact-field shortcode processor + * We use an object method here instead of a static Grunion_Contact_Form_Field class method to parse contact-field shortcodes so that we can tie them to the contact-form object. + * + * @param array $attributes Key => Value pairs as parsed by shortcode_parse_atts() + * @param string|null $content The shortcode's inner content: [contact-field]$content[/contact-field] + * @return HTML for the contact form field + */ + static function parse_contact_field( $attributes, $content ) { + // Don't try to parse contact form fields if not inside a contact form + if ( ! Grunion_Contact_Form_Plugin::$using_contact_form_field ) { + $att_strs = array(); + if ( ! isset( $attributes['label'] ) ) { + $type = isset( $attributes['type'] ) ? $attributes['type'] : null; + $attributes['label'] = self::get_default_label_from_type( $type ); + } + foreach ( $attributes as $att => $val ) { + if ( is_numeric( $att ) ) { // Is a valueless attribute + $att_strs[] = esc_html( $val ); + } elseif ( isset( $val ) ) { // A regular attr - value pair + if ( ( $att === 'options' || $att === 'values' ) && is_string( $val ) ) { // remove any empty strings + $val = explode( ',', $val ); + } + if ( is_array( $val ) ) { + $val = array_filter( $val, array( __CLASS__, 'remove_empty' ) ); // removes any empty strings + $att_strs[] = esc_html( $att ) . '="' . implode( ',', array_map( 'esc_html', $val ) ) . '"'; + } elseif ( is_bool( $val ) ) { + $att_strs[] = esc_html( $att ) . '="' . esc_html( $val ? '1' : '' ) . '"'; + } else { + $att_strs[] = esc_html( $att ) . '="' . esc_html( $val ) . '"'; + } + } + } + + $html = '[contact-field ' . implode( ' ', $att_strs ); + + if ( isset( $content ) && ! empty( $content ) ) { // If there is content, let's add a closing tag + $html .= ']' . esc_html( $content ) . '[/contact-field]'; + } else { // Otherwise let's add a closing slash in the first tag + $html .= '/]'; + } + + return $html; + } + + $form = Grunion_Contact_Form::$current_form; + + $field = new Grunion_Contact_Form_Field( $attributes, $content, $form ); + + $field_id = $field->get_attribute( 'id' ); + if ( $field_id ) { + $form->fields[ $field_id ] = $field; + } else { + $form->fields[] = $field; + } + + if ( + isset( $_POST['action'] ) && 'grunion-contact-form' === $_POST['action'] + && + isset( $_POST['contact-form-id'] ) && $form->get_attribute( 'id' ) == $_POST['contact-form-id'] + && + isset( $_POST['contact-form-hash'] ) && hash_equals( $form->hash, $_POST['contact-form-hash'] ) + ) { + // If we're processing a POST submission for this contact form, validate the field value so we can show errors as necessary. + $field->validate(); + } + + // Output HTML + return $field->render(); + } + + static function get_default_label_from_type( $type ) { + switch ( $type ) { + case 'text': + return __( 'Text', 'jetpack' ); + case 'name': + return __( 'Name', 'jetpack' ); + case 'email': + return __( 'Email', 'jetpack' ); + case 'url': + return __( 'Website', 'jetpack' ); + case 'date': + return __( 'Date', 'jetpack' ); + case 'telephone': + return __( 'Phone', 'jetpack' ); + case 'textarea': + return __( 'Message', 'jetpack' ); + case 'checkbox': + return __( 'Checkbox', 'jetpack' ); + case 'checkbox-multiple': + return __( 'Choose several', 'jetpack' ); + case 'radio': + return __( 'Choose one', 'jetpack' ); + case 'select': + return __( 'Select one', 'jetpack' ); + default: + return null; + } + } + + /** + * Loops through $this->fields to generate a (structured) list of field IDs. + * + * Important: Currently the whitelisted fields are defined as follows: + * `name`, `email`, `url`, `subject`, `textarea` + * + * If you need to add new fields to the Contact Form, please don't add them + * to the whitelisted fields and leave them as extra fields. + * + * The reasoning behind this is that both the admin Feedback view and the CSV + * export will not include any fields that are added to the list of + * whitelisted fields without taking proper care to add them to all the + * other places where they accessed/used/saved. + * + * The safest way to add new fields is to add them to the dropdown and the + * HTML list ( @see Grunion_Contact_Form_Field::render ) and don't add them + * to the list of whitelisted fields. This way they will become a part of the + * `extra fields` which are saved in the post meta and will be properly + * handled by the admin Feedback view and the CSV Export without any extra + * work. + * + * If there is need to add a field to the whitelisted fields, then please + * take proper care to add logic to handle the field in the following places: + * + * - Below in the switch statement - so the field is recognized as whitelisted. + * + * - Grunion_Contact_Form::process_submission - validation and logic. + * + * - Grunion_Contact_Form::process_submission - add the field as an additional + * field in the `post_content` when saving the feedback content. + * + * - Grunion_Contact_Form_Plugin::parse_fields_from_content - add mapping + * for the field, defined in the above method. + * + * - Grunion_Contact_Form_Plugin::map_parsed_field_contents_of_post_to_field_names - + * add mapping of the field for the CSV Export. Otherwise it will be missing + * from the exported data. + * + * - admin.php / grunion_manage_post_columns - add the field to the render logic. + * Otherwise it will be missing from the admin Feedback view. + * + * @return array + */ + function get_field_ids() { + $field_ids = array( + 'all' => array(), // array of all field_ids + 'extra' => array(), // array of all non-whitelisted field IDs + + // Whitelisted "standard" field IDs: + // 'email' => field_id, + // 'name' => field_id, + // 'url' => field_id, + // 'subject' => field_id, + // 'textarea' => field_id, + ); + + foreach ( $this->fields as $id => $field ) { + $field_ids['all'][] = $id; + + $type = $field->get_attribute( 'type' ); + if ( isset( $field_ids[ $type ] ) ) { + // This type of field is already present in our whitelist of "standard" fields for this form + // Put it in extra + $field_ids['extra'][] = $id; + continue; + } + + /** + * See method description before modifying the switch cases. + */ + switch ( $type ) { + case 'email': + case 'name': + case 'url': + case 'subject': + case 'textarea': + $field_ids[ $type ] = $id; + break; + default: + // Put everything else in extra + $field_ids['extra'][] = $id; + } + } + + return $field_ids; + } + + /** + * Process the contact form's POST submission + * Stores feedback. Sends email. + */ + function process_submission() { + global $post; + + $plugin = Grunion_Contact_Form_Plugin::init(); + + $id = $this->get_attribute( 'id' ); + $to = $this->get_attribute( 'to' ); + $widget = $this->get_attribute( 'widget' ); + + $contact_form_subject = $this->get_attribute( 'subject' ); + + $to = str_replace( ' ', '', $to ); + $emails = explode( ',', $to ); + + $valid_emails = array(); + + foreach ( (array) $emails as $email ) { + if ( ! is_email( $email ) ) { + continue; + } + + if ( function_exists( 'is_email_address_unsafe' ) && is_email_address_unsafe( $email ) ) { + continue; + } + + $valid_emails[] = $email; + } + + // No one to send it to, which means none of the "to" attributes are valid emails. + // Use default email instead. + if ( ! $valid_emails ) { + $valid_emails = $this->defaults['to']; + } + + $to = $valid_emails; + + // Last ditch effort to set a recipient if somehow none have been set. + if ( empty( $to ) ) { + $to = get_option( 'admin_email' ); + } + + // Make sure we're processing the form we think we're processing... probably a redundant check. + if ( $widget ) { + if ( 'widget-' . $widget != $_POST['contact-form-id'] ) { + return false; + } + } else { + if ( $post->ID != $_POST['contact-form-id'] ) { + return false; + } + } + + $field_ids = $this->get_field_ids(); + + // Initialize all these "standard" fields to null + $comment_author_email = $comment_author_email_label = // v + $comment_author = $comment_author_label = // v + $comment_author_url = $comment_author_url_label = // v + $comment_content = $comment_content_label = null; + + // For each of the "standard" fields, grab their field label and value. + if ( isset( $field_ids['name'] ) ) { + $field = $this->fields[ $field_ids['name'] ]; + $comment_author = Grunion_Contact_Form_Plugin::strip_tags( + stripslashes( + /** This filter is already documented in core/wp-includes/comment-functions.php */ + apply_filters( 'pre_comment_author_name', addslashes( $field->value ) ) + ) + ); + $comment_author_label = Grunion_Contact_Form_Plugin::strip_tags( $field->get_attribute( 'label' ) ); + } + + if ( isset( $field_ids['email'] ) ) { + $field = $this->fields[ $field_ids['email'] ]; + $comment_author_email = Grunion_Contact_Form_Plugin::strip_tags( + stripslashes( + /** This filter is already documented in core/wp-includes/comment-functions.php */ + apply_filters( 'pre_comment_author_email', addslashes( $field->value ) ) + ) + ); + $comment_author_email_label = Grunion_Contact_Form_Plugin::strip_tags( $field->get_attribute( 'label' ) ); + } + + if ( isset( $field_ids['url'] ) ) { + $field = $this->fields[ $field_ids['url'] ]; + $comment_author_url = Grunion_Contact_Form_Plugin::strip_tags( + stripslashes( + /** This filter is already documented in core/wp-includes/comment-functions.php */ + apply_filters( 'pre_comment_author_url', addslashes( $field->value ) ) + ) + ); + if ( 'http://' == $comment_author_url ) { + $comment_author_url = ''; + } + $comment_author_url_label = Grunion_Contact_Form_Plugin::strip_tags( $field->get_attribute( 'label' ) ); + } + + if ( isset( $field_ids['textarea'] ) ) { + $field = $this->fields[ $field_ids['textarea'] ]; + $comment_content = trim( Grunion_Contact_Form_Plugin::strip_tags( $field->value ) ); + $comment_content_label = Grunion_Contact_Form_Plugin::strip_tags( $field->get_attribute( 'label' ) ); + } + + if ( isset( $field_ids['subject'] ) ) { + $field = $this->fields[ $field_ids['subject'] ]; + if ( $field->value ) { + $contact_form_subject = Grunion_Contact_Form_Plugin::strip_tags( $field->value ); + } + } + + $all_values = $extra_values = array(); + $i = 1; // Prefix counter for stored metadata + + // For all fields, grab label and value + foreach ( $field_ids['all'] as $field_id ) { + $field = $this->fields[ $field_id ]; + $label = $i . '_' . $field->get_attribute( 'label' ); + $value = $field->value; + + $all_values[ $label ] = $value; + $i++; // Increment prefix counter for the next field + } + + // For the "non-standard" fields, grab label and value + // Extra fields have their prefix starting from count( $all_values ) + 1 + foreach ( $field_ids['extra'] as $field_id ) { + $field = $this->fields[ $field_id ]; + $label = $i . '_' . $field->get_attribute( 'label' ); + $value = $field->value; + + if ( is_array( $value ) ) { + $value = implode( ', ', $value ); + } + + $extra_values[ $label ] = $value; + $i++; // Increment prefix counter for the next extra field + } + + $contact_form_subject = trim( $contact_form_subject ); + + $comment_author_IP = Grunion_Contact_Form_Plugin::get_ip_address(); + + $vars = array( 'comment_author', 'comment_author_email', 'comment_author_url', 'contact_form_subject', 'comment_author_IP' ); + foreach ( $vars as $var ) { + $$var = str_replace( array( "\n", "\r" ), '', $$var ); + } + + // Ensure that Akismet gets all of the relevant information from the contact form, + // not just the textarea field and predetermined subject. + $akismet_vars = compact( $vars ); + $akismet_vars['comment_content'] = $comment_content; + + foreach ( array_merge( $field_ids['all'], $field_ids['extra'] ) as $field_id ) { + $field = $this->fields[ $field_id ]; + + // Skip any fields that are just a choice from a pre-defined list. They wouldn't have any value + // from a spam-filtering point of view. + if ( in_array( $field->get_attribute( 'type' ), array( 'select', 'checkbox', 'checkbox-multiple', 'radio' ) ) ) { + continue; + } + + // Normalize the label into a slug. + $field_slug = trim( // Strip all leading/trailing dashes. + preg_replace( // Normalize everything to a-z0-9_- + '/[^a-z0-9_]+/', + '-', + strtolower( $field->get_attribute( 'label' ) ) // Lowercase + ), + '-' + ); + + $field_value = ( is_array( $field->value ) ) ? trim( implode( ', ', $field->value ) ) : trim( $field->value ); + + // Skip any values that are already in the array we're sending. + if ( $field_value && in_array( $field_value, $akismet_vars ) ) { + continue; + } + + $akismet_vars[ 'contact_form_field_' . $field_slug ] = $field_value; + } + + $spam = ''; + $akismet_values = $plugin->prepare_for_akismet( $akismet_vars ); + + // Is it spam? + /** This filter is already documented in modules/contact-form/admin.php */ + $is_spam = apply_filters( 'jetpack_contact_form_is_spam', false, $akismet_values ); + if ( is_wp_error( $is_spam ) ) { // WP_Error to abort + return $is_spam; // abort + } elseif ( $is_spam === true ) { // TRUE to flag a spam + $spam = '***SPAM*** '; + } + + if ( ! $comment_author ) { + $comment_author = $comment_author_email; + } + + /** + * Filter the email where a submitted feedback is sent. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param string|array $to Array of valid email addresses, or single email address. + */ + $to = (array) apply_filters( 'contact_form_to', $to ); + $reply_to_addr = $to[0]; // get just the address part before the name part is added + + foreach ( $to as $to_key => $to_value ) { + $to[ $to_key ] = Grunion_Contact_Form_Plugin::strip_tags( $to_value ); + $to[ $to_key ] = self::add_name_to_address( $to_value ); + } + + $blog_url = parse_url( site_url() ); + $from_email_addr = 'wordpress@' . $blog_url['host']; + + if ( ! empty( $comment_author_email ) ) { + $reply_to_addr = $comment_author_email; + } + + $headers = 'From: "' . $comment_author . '" <' . $from_email_addr . ">\r\n" . + 'Reply-To: "' . $comment_author . '" <' . $reply_to_addr . ">\r\n"; + + // Build feedback reference + $feedback_time = current_time( 'mysql' ); + $feedback_title = "{$comment_author} - {$feedback_time}"; + $feedback_id = md5( $feedback_title ); + + $all_values = array_merge( + $all_values, array( + 'entry_title' => the_title_attribute( 'echo=0' ), + 'entry_permalink' => esc_url( get_permalink( get_the_ID() ) ), + 'feedback_id' => $feedback_id, + ) + ); + + /** This filter is already documented in modules/contact-form/admin.php */ + $subject = apply_filters( 'contact_form_subject', $contact_form_subject, $all_values ); + $url = $widget ? home_url( '/' ) : get_permalink( $post->ID ); + + $date_time_format = _x( '%1$s \a\t %2$s', '{$date_format} \a\t {$time_format}', 'jetpack' ); + $date_time_format = sprintf( $date_time_format, get_option( 'date_format' ), get_option( 'time_format' ) ); + $time = date_i18n( $date_time_format, current_time( 'timestamp' ) ); + + // keep a copy of the feedback as a custom post type + $feedback_status = $is_spam === true ? 'spam' : 'publish'; + + foreach ( (array) $akismet_values as $av_key => $av_value ) { + $akismet_values[ $av_key ] = Grunion_Contact_Form_Plugin::strip_tags( $av_value ); + } + + foreach ( (array) $all_values as $all_key => $all_value ) { + $all_values[ $all_key ] = Grunion_Contact_Form_Plugin::strip_tags( $all_value ); + } + + foreach ( (array) $extra_values as $ev_key => $ev_value ) { + $extra_values[ $ev_key ] = Grunion_Contact_Form_Plugin::strip_tags( $ev_value ); + } + + /* + We need to make sure that the post author is always zero for contact + * form submissions. This prevents export/import from trying to create + * new users based on form submissions from people who were logged in + * at the time. + * + * Unfortunately wp_insert_post() tries very hard to make sure the post + * author gets the currently logged in user id. That is how we ended up + * with this work around. */ + add_filter( 'wp_insert_post_data', array( $plugin, 'insert_feedback_filter' ), 10, 2 ); + + $post_id = wp_insert_post( + array( + 'post_date' => addslashes( $feedback_time ), + 'post_type' => 'feedback', + 'post_status' => addslashes( $feedback_status ), + 'post_parent' => (int) $post->ID, + 'post_title' => addslashes( wp_kses( $feedback_title, array() ) ), + 'post_content' => addslashes( wp_kses( $comment_content . "\n<!--more-->\n" . "AUTHOR: {$comment_author}\nAUTHOR EMAIL: {$comment_author_email}\nAUTHOR URL: {$comment_author_url}\nSUBJECT: {$subject}\nIP: {$comment_author_IP}\n" . @print_r( $all_values, true ), array() ) ), // so that search will pick up this data + 'post_name' => $feedback_id, + ) + ); + + // once insert has finished we don't need this filter any more + remove_filter( 'wp_insert_post_data', array( $plugin, 'insert_feedback_filter' ), 10 ); + + update_post_meta( $post_id, '_feedback_extra_fields', $this->addslashes_deep( $extra_values ) ); + + if ( 'publish' == $feedback_status ) { + // Increase count of unread feedback. + $unread = get_option( 'feedback_unread_count', 0 ) + 1; + update_option( 'feedback_unread_count', $unread ); + } + + if ( defined( 'AKISMET_VERSION' ) ) { + update_post_meta( $post_id, '_feedback_akismet_values', $this->addslashes_deep( $akismet_values ) ); + } + + $message = self::get_compiled_form( $post_id, $this ); + + array_push( + $message, + '<br />', + '<hr />', + __( 'Time:', 'jetpack' ) . ' ' . $time . '<br />', + __( 'IP Address:', 'jetpack' ) . ' ' . $comment_author_IP . '<br />', + __( 'Contact Form URL:', 'jetpack' ) . ' ' . $url . '<br />' + ); + + if ( is_user_logged_in() ) { + array_push( + $message, + sprintf( + '<p>' . __( 'Sent by a verified %s user.', 'jetpack' ) . '</p>', + isset( $GLOBALS['current_site']->site_name ) && $GLOBALS['current_site']->site_name ? + $GLOBALS['current_site']->site_name : '"' . get_option( 'blogname' ) . '"' + ) + ); + } else { + array_push( $message, '<p>' . __( 'Sent by an unverified visitor to your site.', 'jetpack' ) . '</p>' ); + } + + $message = join( $message, '' ); + + /** + * Filters the message sent via email after a successful form submission. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param string $message Feedback email message. + */ + $message = apply_filters( 'contact_form_message', $message ); + + // This is called after `contact_form_message`, in order to preserve back-compat + $message = self::wrap_message_in_html_tags( $message ); + + update_post_meta( $post_id, '_feedback_email', $this->addslashes_deep( compact( 'to', 'message' ) ) ); + + /** + * Fires right before the contact form message is sent via email to + * the recipient specified in the contact form. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param integer $post_id Post contact form lives on + * @param array $all_values Contact form fields + * @param array $extra_values Contact form fields not included in $all_values + */ + do_action( 'grunion_pre_message_sent', $post_id, $all_values, $extra_values ); + + // schedule deletes of old spam feedbacks + if ( ! wp_next_scheduled( 'grunion_scheduled_delete' ) ) { + wp_schedule_event( time() + 250, 'daily', 'grunion_scheduled_delete' ); + } + + if ( + $is_spam !== true && + /** + * Filter to choose whether an email should be sent after each successful contact form submission. + * + * @module contact-form + * + * @since 2.6.0 + * + * @param bool true Should an email be sent after a form submission. Default to true. + * @param int $post_id Post ID. + */ + true === apply_filters( 'grunion_should_send_email', true, $post_id ) + ) { + self::wp_mail( $to, "{$spam}{$subject}", $message, $headers ); + } elseif ( + true === $is_spam && + /** + * Choose whether an email should be sent for each spam contact form submission. + * + * @module contact-form + * + * @since 1.3.1 + * + * @param bool false Should an email be sent after a spam form submission. Default to false. + */ + apply_filters( 'grunion_still_email_spam', false ) == true + ) { // don't send spam by default. Filterable. + self::wp_mail( $to, "{$spam}{$subject}", $message, $headers ); + } + + /** + * Fires an action hook right after the email(s) have been sent. + * + * @module contact-form + * + * @since 7.3.0 + * + * @param int $post_id Post contact form lives on. + * @param string|array $to Array of valid email addresses, or single email address. + * @param string $subject Feedback email subject. + * @param string $message Feedback email message. + * @param string|array $headers Optional. Additional headers. + * @param array $all_values Contact form fields. + * @param array $extra_values Contact form fields not included in $all_values + */ + do_action( 'grunion_after_message_sent', $post_id, $to, $subject, $message, $headers, $all_values, $extra_values ); + + if ( defined( 'DOING_AJAX' ) && DOING_AJAX ) { + return self::success_message( $post_id, $this ); + } + + $redirect = wp_get_referer(); + if ( ! $redirect ) { // wp_get_referer() returns false if the referer is the same as the current page + $redirect = $_SERVER['REQUEST_URI']; + } + + $redirect = add_query_arg( + urlencode_deep( + array( + 'contact-form-id' => $id, + 'contact-form-sent' => $post_id, + 'contact-form-hash' => $this->hash, + '_wpnonce' => wp_create_nonce( "contact-form-sent-{$post_id}" ), // wp_nonce_url HTMLencodes :( + ) + ), $redirect + ); + + /** + * Filter the URL where the reader is redirected after submitting a form. + * + * @module contact-form + * + * @since 1.9.0 + * + * @param string $redirect Post submission URL. + * @param int $id Contact Form ID. + * @param int $post_id Post ID. + */ + $redirect = apply_filters( 'grunion_contact_form_redirect_url', $redirect, $id, $post_id ); + + wp_safe_redirect( $redirect ); + exit; + } + + /** + * Wrapper for wp_mail() that enables HTML messages with text alternatives + * + * @param string|array $to Array or comma-separated list of email addresses to send message. + * @param string $subject Email subject. + * @param string $message Message contents. + * @param string|array $headers Optional. Additional headers. + * @param string|array $attachments Optional. Files to attach. + * + * @return bool Whether the email contents were sent successfully. + */ + public static function wp_mail( $to, $subject, $message, $headers = '', $attachments = array() ) { + add_filter( 'wp_mail_content_type', __CLASS__ . '::get_mail_content_type' ); + add_action( 'phpmailer_init', __CLASS__ . '::add_plain_text_alternative' ); + + $result = wp_mail( $to, $subject, $message, $headers, $attachments ); + + remove_filter( 'wp_mail_content_type', __CLASS__ . '::get_mail_content_type' ); + remove_action( 'phpmailer_init', __CLASS__ . '::add_plain_text_alternative' ); + + return $result; + } + + /** + * Add a display name part to an email address + * + * SpamAssassin doesn't like addresses in HTML messages that are missing display names (e.g., `foo@bar.org` + * instead of `"Foo Bar" <foo@bar.org>`. + * + * @param string $address + * + * @return string + */ + function add_name_to_address( $address ) { + // If it's just the address, without a display name + if ( is_email( $address ) ) { + $address_parts = explode( '@', $address ); + $address = sprintf( '"%s" <%s>', $address_parts[0], $address ); + } + + return $address; + } + + /** + * Get the content type that should be assigned to outbound emails + * + * @return string + */ + static function get_mail_content_type() { + return 'text/html'; + } + + /** + * Wrap a message body with the appropriate in HTML tags + * + * This helps to ensure correct parsing by clients, and also helps avoid triggering spam filtering rules + * + * @param string $body + * + * @return string + */ + static function wrap_message_in_html_tags( $body ) { + // Don't do anything if the message was already wrapped in HTML tags + // That could have be done by a plugin via filters + if ( false !== strpos( $body, '<html' ) ) { + return $body; + } + + $html_message = sprintf( + // The tabs are just here so that the raw code is correctly formatted for developers + // They're removed so that they don't affect the final message sent to users + str_replace( + "\t", '', + '<!doctype html> + <html xmlns="http://www.w3.org/1999/xhtml"> + <body> + + %s + + </body> + </html>' + ), + $body + ); + + return $html_message; + } + + /** + * Add a plain-text alternative part to an outbound email + * + * This makes the message more accessible to mail clients that aren't HTML-aware, and decreases the likelihood + * that the message will be flagged as spam. + * + * @param PHPMailer $phpmailer + */ + static function add_plain_text_alternative( $phpmailer ) { + // Add an extra break so that the extra space above the <p> is preserved after the <p> is stripped out + $alt_body = str_replace( '<p>', '<p><br />', $phpmailer->Body ); + + // Convert <br> to \n breaks, to preserve the space between lines that we want to keep + $alt_body = str_replace( array( '<br>', '<br />' ), "\n", $alt_body ); + + // Convert <hr> to an plain-text equivalent, to preserve the integrity of the message + $alt_body = str_replace( array( '<hr>', '<hr />' ), "----\n", $alt_body ); + + // Trim the plain text message to remove the \n breaks that were after <doctype>, <html>, and <body> + $phpmailer->AltBody = trim( strip_tags( $alt_body ) ); + } + + function addslashes_deep( $value ) { + if ( is_array( $value ) ) { + return array_map( array( $this, 'addslashes_deep' ), $value ); + } elseif ( is_object( $value ) ) { + $vars = get_object_vars( $value ); + foreach ( $vars as $key => $data ) { + $value->{$key} = $this->addslashes_deep( $data ); + } + return $value; + } + + return addslashes( $value ); + } +} + +/** + * Class for the contact-field shortcode. + * Parses shortcode to output the contact form field as HTML. + * Validates input. + */ +class Grunion_Contact_Form_Field extends Crunion_Contact_Form_Shortcode { + public $shortcode_name = 'contact-field'; + + /** + * @var Grunion_Contact_Form parent form + */ + public $form; + + /** + * @var string default or POSTed value + */ + public $value; + + /** + * @var bool Is the input invalid? + */ + public $error = false; + + /** + * @param array $attributes An associative array of shortcode attributes. @see shortcode_atts() + * @param null|string $content Null for selfclosing shortcodes. The inner content otherwise. + * @param Grunion_Contact_Form $form The parent form + */ + function __construct( $attributes, $content = null, $form = null ) { + $attributes = shortcode_atts( + array( + 'label' => null, + 'type' => 'text', + 'required' => false, + 'options' => array(), + 'id' => null, + 'default' => null, + 'values' => null, + 'placeholder' => null, + 'class' => null, + ), $attributes, 'contact-field' + ); + + // special default for subject field + if ( 'subject' == $attributes['type'] && is_null( $attributes['default'] ) && ! is_null( $form ) ) { + $attributes['default'] = $form->get_attribute( 'subject' ); + } + + // allow required=1 or required=true + if ( '1' == $attributes['required'] || 'true' == strtolower( $attributes['required'] ) ) { + $attributes['required'] = true; + } else { + $attributes['required'] = false; + } + + // parse out comma-separated options list (for selects, radios, and checkbox-multiples) + if ( ! empty( $attributes['options'] ) && is_string( $attributes['options'] ) ) { + $attributes['options'] = array_map( 'trim', explode( ',', $attributes['options'] ) ); + + if ( ! empty( $attributes['values'] ) && is_string( $attributes['values'] ) ) { + $attributes['values'] = array_map( 'trim', explode( ',', $attributes['values'] ) ); + } + } + + if ( $form ) { + // make a unique field ID based on the label, with an incrementing number if needed to avoid clashes + $form_id = $form->get_attribute( 'id' ); + $id = isset( $attributes['id'] ) ? $attributes['id'] : false; + + $unescaped_label = $this->unesc_attr( $attributes['label'] ); + $unescaped_label = str_replace( '%', '-', $unescaped_label ); // jQuery doesn't like % in IDs? + $unescaped_label = preg_replace( '/[^a-zA-Z0-9.-_:]/', '', $unescaped_label ); + + if ( empty( $id ) ) { + $id = sanitize_title_with_dashes( 'g' . $form_id . '-' . $unescaped_label ); + $i = 0; + $max_tries = 99; + while ( isset( $form->fields[ $id ] ) ) { + $i++; + $id = sanitize_title_with_dashes( 'g' . $form_id . '-' . $unescaped_label . '-' . $i ); + + if ( $i > $max_tries ) { + break; + } + } + } + + $attributes['id'] = $id; + } + + parent::__construct( $attributes, $content ); + + // Store parent form + $this->form = $form; + } + + /** + * This field's input is invalid. Flag as invalid and add an error to the parent form + * + * @param string $message The error message to display on the form. + */ + function add_error( $message ) { + $this->is_error = true; + + if ( ! is_wp_error( $this->form->errors ) ) { + $this->form->errors = new WP_Error; + } + + $this->form->errors->add( $this->get_attribute( 'id' ), $message ); + } + + /** + * Is the field input invalid? + * + * @see $error + * + * @return bool + */ + function is_error() { + return $this->error; + } + + /** + * Validates the form input + */ + function validate() { + // If it's not required, there's nothing to validate + if ( ! $this->get_attribute( 'required' ) ) { + return; + } + + $field_id = $this->get_attribute( 'id' ); + $field_type = $this->get_attribute( 'type' ); + $field_label = $this->get_attribute( 'label' ); + + if ( isset( $_POST[ $field_id ] ) ) { + if ( is_array( $_POST[ $field_id ] ) ) { + $field_value = array_map( 'stripslashes', $_POST[ $field_id ] ); + } else { + $field_value = stripslashes( $_POST[ $field_id ] ); + } + } else { + $field_value = ''; + } + + switch ( $field_type ) { + case 'email': + // Make sure the email address is valid + if ( ! is_email( $field_value ) ) { + /* translators: %s is the name of a form field */ + $this->add_error( sprintf( __( '%s requires a valid email address', 'jetpack' ), $field_label ) ); + } + break; + case 'checkbox-multiple': + // Check that there is at least one option selected + if ( empty( $field_value ) ) { + /* translators: %s is the name of a form field */ + $this->add_error( sprintf( __( '%s requires at least one selection', 'jetpack' ), $field_label ) ); + } + break; + default: + // Just check for presence of any text + if ( ! strlen( trim( $field_value ) ) ) { + /* translators: %s is the name of a form field */ + $this->add_error( sprintf( __( '%s is required', 'jetpack' ), $field_label ) ); + } + } + } + + + /** + * Check the default value for options field + * + * @param string value + * @param int index + * @param string default value + * + * @return string + */ + public function get_option_value( $value, $index, $options ) { + if ( empty( $value[ $index ] ) ) { + return $options; + } + return $value[ $index ]; + } + + /** + * Outputs the HTML for this form field + * + * @return string HTML + */ + function render() { + global $current_user, $user_identity; + + $field_id = $this->get_attribute( 'id' ); + $field_type = $this->get_attribute( 'type' ); + $field_label = $this->get_attribute( 'label' ); + $field_required = $this->get_attribute( 'required' ); + $field_placeholder = $this->get_attribute( 'placeholder' ); + $class = 'date' === $field_type ? 'jp-contact-form-date' : $this->get_attribute( 'class' ); + + /** + * Filters the "class" attribute of the contact form input + * + * @module contact-form + * + * @since 6.6.0 + * + * @param string $class Additional CSS classes for input class attribute. + */ + $field_class = apply_filters( 'jetpack_contact_form_input_class', $class ); + + if ( isset( $_POST[ $field_id ] ) ) { + if ( is_array( $_POST[ $field_id ] ) ) { + $this->value = array_map( 'stripslashes', $_POST[ $field_id ] ); + } else { + $this->value = stripslashes( (string) $_POST[ $field_id ] ); + } + } elseif ( isset( $_GET[ $field_id ] ) ) { + $this->value = stripslashes( (string) $_GET[ $field_id ] ); + } elseif ( + is_user_logged_in() && + ( ( defined( 'IS_WPCOM' ) && IS_WPCOM ) || + /** + * Allow third-party tools to prefill the contact form with the user's details when they're logged in. + * + * @module contact-form + * + * @since 3.2.0 + * + * @param bool false Should the Contact Form be prefilled with your details when you're logged in. Default to false. + */ + true === apply_filters( 'jetpack_auto_fill_logged_in_user', false ) + ) + ) { + // Special defaults for logged-in users + switch ( $this->get_attribute( 'type' ) ) { + case 'email': + $this->value = $current_user->data->user_email; + break; + case 'name': + $this->value = $user_identity; + break; + case 'url': + $this->value = $current_user->data->user_url; + break; + default: + $this->value = $this->get_attribute( 'default' ); + } + } else { + $this->value = $this->get_attribute( 'default' ); + } + + $field_value = Grunion_Contact_Form_Plugin::strip_tags( $this->value ); + $field_label = Grunion_Contact_Form_Plugin::strip_tags( $field_label ); + + $rendered_field = $this->render_field( $field_type, $field_id, $field_label, $field_value, $field_class, $field_placeholder, $field_required ); + + /** + * Filter the HTML of the Contact Form. + * + * @module contact-form + * + * @since 2.6.0 + * + * @param string $rendered_field Contact Form HTML output. + * @param string $field_label Field label. + * @param int|null $id Post ID. + */ + return apply_filters( 'grunion_contact_form_field_html', $rendered_field, $field_label, ( in_the_loop() ? get_the_ID() : null ) ); + } + + function render_label( $type = '', $id, $label, $required, $required_field_text ) { + + $type_class = $type ? ' ' .$type : ''; + return + "<label + for='" . esc_attr( $id ) . "' + class='grunion-field-label{$type_class}" . ( $this->is_error() ? ' form-error' : '' ) . "' + >" + . esc_html( $label ) + . ( $required ? '<span>' . $required_field_text . '</span>' : '' ) + . "</label>\n"; + + } + + function render_input_field( $type, $id, $value, $class, $placeholder, $required ) { + return "<input + type='". esc_attr( $type ) ."' + name='" . esc_attr( $id ) . "' + id='" . esc_attr( $id ) . "' + value='" . esc_attr( $value ) . "' + " . $class . $placeholder . ' + ' . ( $required ? "required aria-required='true'" : '' ) . " + />\n"; + } + + function render_email_field( $id, $label, $value, $class, $required, $required_field_text, $placeholder ) { + $field = $this->render_label( 'email', $id, $label, $required, $required_field_text ); + $field .= $this->render_input_field( 'email', $id, $value, $class, $placeholder, $required ); + return $field; + } + + function render_telephone_field( $id, $label, $value, $class, $required, $required_field_text, $placeholder ) { + $field = $this->render_label( 'telephone', $id, $label, $required, $required_field_text ); + $field .= $this->render_input_field( 'tel', $id, $value, $class, $placeholder, $required ); + return $field; + } + + function render_url_field( $id, $label, $value, $class, $required, $required_field_text, $placeholder ) { + $field = $this->render_label( 'url', $id, $label, $required, $required_field_text ); + $field .= $this->render_input_field( 'url', $id, $value, $class, $placeholder, $required ); + return $field; + } + + function render_textarea_field( $id, $label, $value, $class, $required, $required_field_text, $placeholder ) { + $field = $this->render_label( 'textarea', 'contact-form-comment-' . $id, $label, $required, $required_field_text ); + $field .= "<textarea + name='" . esc_attr( $id ) . "' + id='contact-form-comment-" . esc_attr( $id ) . "' + rows='20' " + . $class + . $placeholder + . ' ' . ( $required ? "required aria-required='true'" : '' ) . + '>' . esc_textarea( $value ) + . "</textarea>\n"; + return $field; + } + + function render_radio_field( $id, $label, $value, $class, $required, $required_field_text ) { + $field = $this->render_label( '', $id, $label, $required, $required_field_text ); + foreach ( (array) $this->get_attribute( 'options' ) as $optionIndex => $option ) { + $option = Grunion_Contact_Form_Plugin::strip_tags( $option ); + if ( $option ) { + $field .= "\t\t<label class='grunion-radio-label radio" . ( $this->is_error() ? ' form-error' : '' ) . "'>"; + $field .= "<input + type='radio' + name='" . esc_attr( $id ) . "' + value='" . esc_attr( $this->get_option_value( $this->get_attribute( 'values' ), $optionIndex, $option ) ) . "' " + . $class + . checked( $option, $value, false ) . ' ' + . ( $required ? "required aria-required='true'" : '' ) + . '/> '; + $field .= esc_html( $option ) . "</label>\n"; + $field .= "\t\t<div class='clear-form'></div>\n"; + } + } + return $field; + } + + function render_checkbox_field( $id, $label, $value, $class, $required, $required_field_text ) { + $field = "<label class='grunion-field-label checkbox" . ( $this->is_error() ? ' form-error' : '' ) . "'>"; + $field .= "\t\t<input type='checkbox' name='" . esc_attr( $id ) . "' value='" . esc_attr__( 'Yes', 'jetpack' ) . "' " . $class . checked( (bool) $value, true, false ) . ' ' . ( $required ? "required aria-required='true'" : '' ) . "/> \n"; + $field .= "\t\t" . esc_html( $label ) . ( $required ? '<span>' . $required_field_text . '</span>' : '' ); + $field .= "</label>\n"; + $field .= "<div class='clear-form'></div>\n"; + return $field; + } + + function render_checkbox_multiple_field( $id, $label, $value, $class, $required, $required_field_text ) { + $field = $this->render_label( '', $id, $label, $required, $required_field_text ); + foreach ( (array) $this->get_attribute( 'options' ) as $optionIndex => $option ) { + $option = Grunion_Contact_Form_Plugin::strip_tags( $option ); + if ( $option ) { + $field .= "\t\t<label class='grunion-checkbox-multiple-label checkbox-multiple" . ( $this->is_error() ? ' form-error' : '' ) . "'>"; + $field .= "<input type='checkbox' name='" . esc_attr( $id ) . "[]' value='" . esc_attr( $this->get_option_value( $this->get_attribute( 'values' ), $optionIndex, $option ) ) . "' " . $class . checked( in_array( $option, (array) $value ), true, false ) . ' /> '; + $field .= esc_html( $option ) . "</label>\n"; + $field .= "\t\t<div class='clear-form'></div>\n"; + } + } + + return $field; + } + + function render_select_field( $id, $label, $value, $class, $required, $required_field_text ) { + $field = $this->render_label( 'select', $id, $label, $required, $required_field_text ); + $field .= "\t<select name='" . esc_attr( $id ) . "' id='" . esc_attr( $id ) . "' " . $class . ( $required ? "required aria-required='true'" : '' ) . ">\n"; + foreach ( (array) $this->get_attribute( 'options' ) as $optionIndex => $option ) { + $option = Grunion_Contact_Form_Plugin::strip_tags( $option ); + if ( $option ) { + $field .= "\t\t<option" + . selected( $option, $value, false ) + . " value='" . esc_attr( $this->get_option_value( $this->get_attribute( 'values' ), $optionIndex, $option ) ) + . "'>" . esc_html( $option ) + . "</option>\n"; + } + } + $field .= "\t</select>\n"; + return $field; + } + + function render_date_field( $id, $label, $value, $class, $required, $required_field_text, $placeholder ) { + $field = $this->render_label( 'date', $id, $label, $required, $required_field_text ); + $field .= $this->render_input_field( 'text', $id, $value, $class, $placeholder, $required ); + + wp_enqueue_script( + 'grunion-frontend', + Jetpack::get_file_url_for_environment( + '_inc/build/contact-form/js/grunion-frontend.min.js', + 'modules/contact-form/js/grunion-frontend.js' + ), + array( 'jquery', 'jquery-ui-datepicker' ) + ); + wp_enqueue_style( 'jp-jquery-ui-datepicker', plugins_url( 'css/jquery-ui-datepicker.css', __FILE__ ), array( 'dashicons' ), '1.0' ); + + // Using Core's built-in datepicker localization routine + wp_localize_jquery_ui_datepicker(); + return $field; + } + + function render_default_field( $id, $label, $value, $class, $required, $required_field_text, $placeholder, $type ) { + $field = $this->render_label( $type, $id, $label, $required, $required_field_text ); + $field .= $this->render_input_field( 'text', $id, $value, $class, $placeholder, $required ); + return $field; + } + + function render_field( $type, $id, $label, $value, $class, $placeholder, $required ) { + + $field_placeholder = ( ! empty( $placeholder ) ) ? "placeholder='" . esc_attr( $placeholder ) . "'" : ''; + $field_class = "class='" . trim( esc_attr( $type ) . ' ' . esc_attr( $class ) ) . "' "; + $wrap_classes = empty( $class ) ? '' : implode( '-wrap ', array_filter( explode( ' ', $class ) ) ) . '-wrap'; // this adds + + $shell_field_class = "class='grunion-field-wrap grunion-field-" . trim( esc_attr( $type ) . '-wrap ' . esc_attr( $wrap_classes ) ) . "' "; + /** + /** + * Filter the Contact Form required field text + * + * @module contact-form + * + * @since 3.8.0 + * + * @param string $var Required field text. Default is "(required)". + */ + $required_field_text = esc_html( apply_filters( 'jetpack_required_field_text', __( '(required)', 'jetpack' ) ) ); + + $field = "\n<div {$shell_field_class} >\n"; // new in Jetpack 6.8.0 + // If they are logged in, and this is their site, don't pre-populate fields + if ( current_user_can( 'manage_options' ) ) { + $value = ''; + } + switch ( $type ) { + case 'email': + $field .= $this->render_email_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder ); + break; + case 'telephone': + $field .= $this->render_telephone_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder ); + break; + case 'url': + $field .= $this->render_url_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder ); + break; + case 'textarea': + $field .= $this->render_textarea_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder ); + break; + case 'radio': + $field .= $this->render_radio_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder ); + break; + case 'checkbox': + $field .= $this->render_checkbox_field( $id, $label, $value, $field_class, $required, $required_field_text ); + break; + case 'checkbox-multiple': + $field .= $this->render_checkbox_multiple_field( $id, $label, $value, $field_class, $required, $required_field_text ); + break; + case 'select': + $field .= $this->render_select_field( $id, $label, $value, $field_class, $required, $required_field_text ); + break; + case 'date': + $field .= $this->render_date_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder ); + break; + default: // text field + $field .= $this->render_default_field( $id, $label, $value, $field_class, $required, $required_field_text, $field_placeholder, $type ); + break; + } + $field .= "\t</div>\n"; + return $field; + } +} + +add_action( 'init', array( 'Grunion_Contact_Form_Plugin', 'init' ), 9 ); + +add_action( 'grunion_scheduled_delete', 'grunion_delete_old_spam' ); + +/** + * Deletes old spam feedbacks to keep the posts table size under control + */ +function grunion_delete_old_spam() { + global $wpdb; + + $grunion_delete_limit = 100; + + $now_gmt = current_time( 'mysql', 1 ); + $sql = $wpdb->prepare( + " + SELECT `ID` + FROM $wpdb->posts + WHERE DATE_SUB( %s, INTERVAL 15 DAY ) > `post_date_gmt` + AND `post_type` = 'feedback' + AND `post_status` = 'spam' + LIMIT %d + ", $now_gmt, $grunion_delete_limit + ); + $post_ids = $wpdb->get_col( $sql ); + + foreach ( (array) $post_ids as $post_id ) { + // force a full delete, skip the trash + wp_delete_post( $post_id, true ); + } + + if ( + /** + * Filter if the module run OPTIMIZE TABLE on the core WP tables. + * + * @module contact-form + * + * @since 1.3.1 + * @since 6.4.0 Set to false by default. + * + * @param bool $filter Should Jetpack optimize the table, defaults to false. + */ + apply_filters( 'grunion_optimize_table', false ) + ) { + $wpdb->query( "OPTIMIZE TABLE $wpdb->posts" ); + } + + // if we hit the max then schedule another run + if ( count( $post_ids ) >= $grunion_delete_limit ) { + wp_schedule_single_event( time() + 700, 'grunion_scheduled_delete' ); + } +} diff --git a/plugins/jetpack/modules/contact-form/grunion-editor-view.php b/plugins/jetpack/modules/contact-form/grunion-editor-view.php new file mode 100644 index 00000000..d1ba0439 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/grunion-editor-view.php @@ -0,0 +1,299 @@ +<?php + +/* + * A prototype to allow inline editing / editor views for contact forms.\ + * + * Originally developed in: http://github.com/automattic/gm2016-grunion-editor + * Authors: Michael Arestad, Andrew Ozz, and George Stephanis + */ + +class Grunion_Editor_View { + + /** + * Add hooks according to screen. + * + * @param WP_Screen $screen Data about current screen. + */ + public static function add_hooks( $screen ) { + if ( isset( $screen->base ) && 'post' === $screen->base ) { + add_action( 'admin_notices', array( __CLASS__, 'handle_editor_view_js' ) ); + add_action( 'admin_head', array( __CLASS__, 'admin_head' ) ); + } + } + + public static function admin_head() { + remove_action( 'media_buttons', 'grunion_media_button', 999 ); + add_action( 'media_buttons', array( __CLASS__, 'grunion_media_button' ), 999 ); + } + + public static function grunion_media_button() { + $title = __( 'Add Contact Form', 'jetpack' ); + ?> + + <button type="button" id="insert-jetpack-contact-form" class="button" title="<?php echo esc_attr( $title ); ?>" href="javascript:;"> + <span class="jetpack-contact-form-icon"></span> + <?php echo esc_html( $title ); ?> + </button> + + <?php + } + + public static function mce_external_plugins( $plugin_array ) { + $plugin_array['grunion_form'] = Jetpack::get_file_url_for_environment( + '_inc/build/contact-form/js/tinymce-plugin-form-button.min.js', + 'modules/contact-form/js/tinymce-plugin-form-button.js' + ); + return $plugin_array; + } + + public static function mce_buttons( $buttons ) { + $size = sizeof( $buttons ); + $buttons1 = array_slice( $buttons, 0, $size - 1 ); + $buttons2 = array_slice( $buttons, $size - 1 ); + return array_merge( + $buttons1, + array( 'grunion' ), + $buttons2 + ); + } + + /** + * WordPress Shortcode Editor View JS Code + */ + public static function handle_editor_view_js() { + add_action( 'admin_print_footer_scripts', array( __CLASS__, 'editor_view_js_templates' ), 1 ); + add_filter( 'mce_external_plugins', array( __CLASS__, 'mce_external_plugins' ) ); + add_filter( 'mce_buttons', array( __CLASS__, 'mce_buttons' ) ); + + wp_enqueue_style( 'grunion-editor-ui', plugins_url( 'css/editor-ui.css', __FILE__ ) ); + wp_style_add_data( 'grunion-editor-ui', 'rtl', 'replace' ); + wp_enqueue_script( + 'grunion-editor-view', + Jetpack::get_file_url_for_environment( + '_inc/build/contact-form/js/editor-view.min.js', + 'modules/contact-form/js/editor-view.js' + ), + array( 'wp-util', 'jquery', 'quicktags' ), + false, + true + ); + wp_localize_script( + 'grunion-editor-view', 'grunionEditorView', array( + 'inline_editing_style' => plugins_url( 'css/editor-inline-editing-style.css', __FILE__ ), + 'inline_editing_style_rtl' => plugins_url( 'css/editor-inline-editing-style-rtl.css', __FILE__ ), + 'dashicons_css_url' => includes_url( 'css/dashicons.css' ), + 'default_form' => '[contact-field label="' . __( 'Name', 'jetpack' ) . '" type="name" required="true" /]' . + '[contact-field label="' . __( 'Email', 'jetpack' ) . '" type="email" required="true" /]' . + '[contact-field label="' . __( 'Website', 'jetpack' ) . '" type="url" /]' . + '[contact-field label="' . __( 'Message', 'jetpack' ) . '" type="textarea" /]', + 'labels' => array( + 'submit_button_text' => __( 'Submit', 'jetpack' ), + /** This filter is documented in modules/contact-form/grunion-contact-form.php */ + 'required_field_text' => apply_filters( 'jetpack_required_field_text', __( '(required)', 'jetpack' ) ), + 'edit_close_ays' => __( 'Are you sure you\'d like to stop editing this form without saving your changes?', 'jetpack' ), + 'quicktags_label' => __( 'contact form', 'jetpack' ), + 'tinymce_label' => __( 'Add contact form', 'jetpack' ), + ), + ) + ); + + add_editor_style( plugin_dir_url( __FILE__ ) . 'css/editor-style.css' ); + } + + /** + * JS Templates. + */ + public static function editor_view_js_templates() { + ?> +<script type="text/html" id="tmpl-grunion-contact-form"> + <form class="card jetpack-contact-form-shortcode-preview" action='#' method='post' class='contact-form commentsblock' onsubmit="return false;"> + {{{ data.body }}} + <p class='contact-submit'> + <input type='submit' value='{{ data.submit_button_text }}' class='pushbutton-wide'/> + </p> + </form> +</script> + +<script type="text/html" id="tmpl-grunion-field-email"> + <div> + <label for='{{ data.id }}' class='grunion-field-label email'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <input type='email' name='{{ data.id }}' id='{{ data.id }}' value='{{ data.value }}' class='{{ data.class }}' placeholder='{{ data.placeholder }}' /> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-telephone"> + <div> + <label for='{{ data.id }}' class='grunion-field-label telephone'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <input type='tel' name='{{ data.id }}' id='{{ data.id }}' value='{{ data.value }}' class='{{ data.class }}' placeholder='{{ data.placeholder }}' /> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-textarea"> + <div> + <label for='contact-form-comment-{{ data.id }}' class='grunion-field-label textarea'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <textarea name='{{ data.id }}' id='contact-form-comment-{{ data.id }}' rows='20' class='{{ data.class }}' placeholder='{{ data.placeholder }}'>{{ data.value }}</textarea> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-radio"> + <div> + <label class='grunion-field-label'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <# _.each( data.options, function( option ) { #> + <label class='grunion-radio-label radio'> + <input type='radio' name='{{ data.id }}' value='{{ option }}' class="{{ data.class }}" <# if ( option === data.value ) print( "checked='checked'" ) #> /> + <span>{{ option }}</span> + </label> + <# }); #> + <div class='clear-form'></div> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-checkbox"> + <div> + <label class='grunion-field-label checkbox'> + <input type='checkbox' name='{{ data.id }}' value='<?php esc_attr__( 'Yes', 'jetpack' ); ?>' class="{{ data.class }}" <# if ( data.value ) print( 'checked="checked"' ) #> /> + <span>{{ data.label }}</span><# if ( data.required ) print( " <span>" + data.required + "</span>" ) #> + </label> + <div class='clear-form'></div> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-checkbox-multiple"> + <div> + <label class='grunion-field-label'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <# _.each( data.options, function( option ) { #> + <label class='grunion-checkbox-multiple-label checkbox-multiple'> + <input type='checkbox' name='{{ data.id }}[]' value='{{ option }}' class="{{ data.class }}" <# if ( option === data.value || _.contains( data.value, option ) ) print( "checked='checked'" ) #> /> + <span>{{ option }}</span> + </label> + <# }); #> + <div class='clear-form'></div> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-select"> + <div> + <label for='{{ data.id }}' class='grunion-field-label select'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <select name='{{ data.id }}' id='{{ data.id }}' class="{{ data.class }}"> + <# _.each( data.options, function( option ) { #> + <option <# if ( option === data.value ) print( "selected='selected'" ) #>>{{ option }}</option> + <# }); #> + </select> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-date"> + <div> + <label for='{{ data.id }}' class='grunion-field-label {{ data.type }}'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <input type='text' name='{{ data.id }}' id='{{ data.id }}' value='{{ data.value }}' class="{{ data.class }}" /> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-text"> + <div> + <label for='{{ data.id }}' class='grunion-field-label {{ data.type }}'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <input type='text' name='{{ data.id }}' id='{{ data.id }}' value='{{ data.value }}' class='{{ data.class }}' placeholder='{{ data.placeholder }}' /> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-url"> + <div> + <label for='{{ data.id }}' class='grunion-field-label {{ data.type }}'>{{ data.label }}<# if ( data.required ) print( " <span>" + data.required + "</span>" ) #></label> + <input type='url' name='{{ data.id }}' id='{{ data.id }}' value='{{ data.value }}' class='{{ data.class }}' placeholder='{{ data.placeholder }}' /> + </div> +</script> + + +<script type="text/html" id="tmpl-grunion-field-edit"> + <div class="card is-compact grunion-field-edit grunion-field-{{ data.type }}" aria-label="<?php esc_attr_e( 'Form Field', 'jetpack' ); ?>"> + <label class="grunion-name"> + <span><?php esc_html_e( 'Field Label', 'jetpack' ); ?></span> + <input type="text" name="label" placeholder="<?php esc_attr_e( 'Label', 'jetpack' ); ?>" value="{{ data.label }}"/> + </label> + + <?php + $grunion_field_types = array( + 'text' => __( 'Text', 'jetpack' ), + 'name' => __( 'Name', 'jetpack' ), + 'email' => __( 'Email', 'jetpack' ), + 'url' => __( 'Website', 'jetpack' ), + 'textarea' => __( 'Textarea', 'jetpack' ), + 'checkbox' => __( 'Checkbox', 'jetpack' ), + 'checkbox-multiple' => __( 'Checkbox with Multiple Items', 'jetpack' ), + 'select' => __( 'Drop down', 'jetpack' ), + 'radio' => __( 'Radio', 'jetpack' ), + 'date' => __( 'Date', 'jetpack' ), + ); + ?> + <div class="grunion-type-options"> + <label class="grunion-type"> + <?php esc_html_e( 'Field Type', 'jetpack' ); ?> + <select name="type"> + <?php foreach ( $grunion_field_types as $type => $label ) : ?> + <option <# if ( '<?php echo esc_js( $type ); ?>' === data.type ) print( "selected='selected'" ) #> value="<?php echo esc_attr( $type ); ?>"> + <?php echo esc_html( $label ); ?> + </option> + <?php endforeach; ?> + </select> + </label> + + <label class="grunion-required"> + <input type="checkbox" name="required" value="1" <# if ( data.required ) print( 'checked="checked"' ) #> /> + <span><?php esc_html_e( 'Required?', 'jetpack' ); ?></span> + </label> + </div> + + <label class="grunion-options"> + <?php esc_html_e( 'Options', 'jetpack' ); ?> + <ol> + <# if ( data.options ) { #> + <# _.each( data.options, function( option ) { #> + <li><input type="text" name="option" value="{{ option }}" /> <a class="delete-option" href="javascript:;"><span class="screen-reader-text"><?php esc_html_e( 'Delete Option', 'jetpack' ); ?></span></a></li> + <# }); #> + <# } else { #> + <li><input type="text" name="option" /> <a class="delete-option" href="javascript:;"><span class="screen-reader-text"><?php esc_html_e( 'Delete Option', 'jetpack' ); ?></span></a></li> + <li><input type="text" name="option" /> <a class="delete-option" href="javascript:;"><span class="screen-reader-text"><?php esc_html_e( 'Delete Option', 'jetpack' ); ?></span></a></li> + <li><input type="text" name="option" /> <a class="delete-option" href="javascript:;"><span class="screen-reader-text"><?php esc_html_e( 'Delete Option', 'jetpack' ); ?></span></a></li> + <# } #> + <li><a class="add-option" href="javascript:;"><?php esc_html_e( 'Add new option...', 'jetpack' ); ?></a></li> + </ol> + </label> + + <a href="javascript:;" class="delete-field"><span class="screen-reader-text"><?php esc_html_e( 'Delete Field', 'jetpack' ); ?></span></a> + </div> +</script> + +<script type="text/html" id="tmpl-grunion-field-edit-option"> + <li><input type="text" name="option" /> <a class="delete-option" href="javascript:;"><span class="screen-reader-text"><?php esc_html_e( 'Delete Option', 'jetpack' ); ?></span></a></li> +</script> + +<script type="text/html" id="tmpl-grunion-editor-inline"> + <h1 id="form-settings-header" class="grunion-section-header"><?php esc_html_e( 'Contact form information', 'jetpack' ); ?></h1> + <section class="card grunion-form-settings" aria-labelledby="form-settings-header"> + <label><?php esc_html_e( 'What would you like the subject of the email to be?', 'jetpack' ); ?> + <input type="text" name="subject" value="{{ data.subject }}" /> + </label> + <label><?php esc_html_e( 'Which email address should we send the submissions to?', 'jetpack' ); ?> + <input type="text" name="to" value="{{ data.to }}" /> + </label> + </section> + <h1 id="form-fields-header" class="grunion-section-header"><?php esc_html_e( 'Contact form fields', 'jetpack' ); ?></h1> + <section class="grunion-fields" aria-labelledby="form-fields-header"> + {{{ data.fields }}} + </section> + <section class="grunion-controls"> + <?php submit_button( esc_html__( 'Add Field', 'jetpack' ), 'secondary', 'add-field', false ); ?> + + <div class="grunion-update-controls"> + <?php submit_button( esc_html__( 'Cancel', 'jetpack' ), 'delete', 'cancel', false ); ?> + <?php submit_button( esc_html__( 'Update Form', 'jetpack' ), 'primary', 'submit', false ); ?> + </div> + </section> +</script> + +</div> + <?php + } +} + +add_action( 'current_screen', array( 'Grunion_Editor_View', 'add_hooks' ) ); diff --git a/plugins/jetpack/modules/contact-form/grunion-form-view.php b/plugins/jetpack/modules/contact-form/grunion-form-view.php new file mode 100644 index 00000000..26ed6dfe --- /dev/null +++ b/plugins/jetpack/modules/contact-form/grunion-form-view.php @@ -0,0 +1,266 @@ +<?php +/** + * Template for form builder + */ + +/** + * Filter to modify the limit of 5 additional contact form fields. + * + * @module contact-form + * + * @since 3.2.0 + * + * @param int 5 Maximum number of additional fields. + */ +$max_new_fields = apply_filters( 'grunion_max_new_fields', 5 ); + +wp_register_script( + 'grunion', + Jetpack::get_file_url_for_environment( + '_inc/build/contact-form/js/grunion.min.js', + 'modules/contact-form/js/grunion.js' + ), + array( 'jquery-ui-sortable', 'jquery-ui-draggable' ), + JETPACK__VERSION +); + +wp_localize_script( + 'grunion', 'GrunionFB_i18n', array( + 'nameLabel' => esc_attr( _x( 'Name', 'Label for HTML form "Name" field in contact form builder', 'jetpack' ) ), + 'emailLabel' => esc_attr( _x( 'Email', 'Label for HTML form "Email" field in contact form builder', 'jetpack' ) ), + 'urlLabel' => esc_attr( _x( 'Website', 'Label for HTML form "URL/Website" field in contact form builder', 'jetpack' ) ), + 'commentLabel' => esc_attr( _x( 'Comment', 'noun', 'jetpack' ) ), + 'newLabel' => esc_attr( _x( 'New Field', 'Default label for new HTML form field in contact form builder', 'jetpack' ) ), + 'optionsLabel' => esc_attr( _x( 'Options', 'Label for the set of options to be included in a user-created dropdown in contact form builder', 'jetpack' ) ), + 'optionsLabel' => esc_attr( _x( 'Option', 'Label for an option to be included in a user-created dropdown in contact form builder', 'jetpack' ) ), + 'firstOptionLabel' => esc_attr( _x( 'First option', 'Default label for the first option to be included in a user-created dropdown in contact form builder', 'jetpack' ) ), + 'problemGeneratingForm' => esc_attr( _x( "Oops, there was a problem generating your form. You'll likely need to try again.", 'error message in contact form builder', 'jetpack' ) ), + 'moveInstructions' => esc_attr__( "Drag up or down\nto re-arrange", 'jetpack' ), + 'moveLabel' => esc_attr( _x( 'move', 'Label to drag HTML form fields around to change their order in contact form builder', 'jetpack' ) ), + 'editLabel' => esc_attr( _x( 'edit', 'Link to edit an HTML form field in contact form builder', 'jetpack' ) ), + 'savedMessage' => esc_attr__( 'Saved successfully', 'jetpack' ), + 'requiredLabel' => esc_attr( _x( '(required)', 'This HTML form field is marked as required by the user in contact form builder', 'jetpack' ) ), + 'exitConfirmMessage' => esc_attr__( 'Are you sure you want to exit the form editor without saving? Any changes you have made will be lost.', 'jetpack' ), + 'maxNewFields' => intval( $max_new_fields ), + ) +); + +?> +<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> +<html xmlns="http://www.w3.org/1999/xhtml"> +<head> +<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> +<title><?php esc_html_e( 'Contact Form', 'jetpack' ); ?></title> +<script type="text/javascript"> + var ajaxurl = '<?php echo admin_url( 'admin-ajax.php' ); ?>'; + var postId = <?php echo absint( $_GET['post_id'] ); ?>; + var ajax_nonce_shortcode = '<?php echo wp_create_nonce( 'grunion_shortcode' ); ?>'; + var ajax_nonce_json = '<?php echo wp_create_nonce( 'grunion_shortcode_to_json' ); ?>'; +</script> +<?php wp_print_scripts( 'grunion' ); ?> +<script type="text/javascript"> + jQuery(document).ready(function () { + FB.ContactForm.init(); + FB.ContactForm.resizePop(); + }); + jQuery(window).resize(function() { + setTimeout(function () { FB.ContactForm.resizePop(); }, 50); + }); +</script> +<style> + /* Reset */ + html { height: 100%; } + body, div, ul, ol, li, h1, h2, h3, h4, h5, h6, form, fieldset, legend, input, button, textarea, p, blockquote, th, td { margin: 0; padding: 0; } + body { background: #F9F9F9; font-family:"Lucida Grande",Verdana,Arial,"Bitstream Vera Sans",sans-serif; font-size:12px; color: #333; line-height:1.5em; height: 100%; width: 100%; padding-bottom: 20px !important; } + a { color: #21759B; text-decoration: none; } + a:hover { text-decoration: underline; text-shadow: none !important; } + h1 { font-size: 21px; color:#5A5A5A; font-family:Georgia,"Times New Roman",Times,serif; font-weight:normal; margin-bottom: 21px; } + h3 { font-size: 13px; color: #666; margin-bottom: 18px; } + input { width: 301px; } + input[type='text'] { padding: 3px 5px; margin-right: 4px; -moz-border-radius:3px; border-radius:3px; -webkit-border-radius:3px; } + input[type='text']:focus { border: 2px solid #80B8D9; outline: 0 !important; } + input[type='checkbox'], input[type='radio'] { width: auto !important; float: left; margin-top: 3px; margin-right: 8px; } + input.fieldError, select.fieldError, textarea.fieldError { border: 2px solid #D56F55; } + img { border: none; } + label { color: #222; font-weight: bold; display: block; margin-bottom: 4px; } + label.radio { width: auto; margin: -2px 0 0 5px; } + label span.label-required { color: #AAA; margin-left: 4px; font-weight: normal; } + td { vertical-align: top; } + select { width: 300px; } + textarea { height: 100px; width: 311px; } + /* Core */ + #media-upload-header { border-bottom: 1px solid #DFDFDF; font-weight:bold; margin:0; padding:3px 5px 0 5px; position:relative; background: #FFF; } + #sidemenu { bottom:-1px; font-size:12px; list-style:none outside none; padding-left:10px; position:relative; left:0; margin:0 5px; overflow:hidden; } + #sidemenu a { text-decoration:none; border-top: 1px solid #FFF; display:block; float:left; line-height:28px; padding:0 13px; outline: none; } + #sidemenu a.current { background-color:#F9F9F9; border-color:#DFDFDF #DFDFDF #F9F9F9; color:#D54E21; -moz-border-radius:4px 4px 0 0; border-radius:4px 4px 0 0; -webkit-border-radius:4px 4px 0 0; border-style:solid; border-width:1px; font-weight:normal; } + #sidemenu li { display:inline; margin-bottom:6px; line-height:200%; list-style:none outside none; margin:0; padding:0; text-align:center; white-space:nowrap; } + .button { background-color:#f2f2f2; border-color:#BBBBBB; min-width:80px; text-align:center; color:#464646; text-shadow:0 1px 0 #FFFFFF; border-style:solid; border-width:1px; cursor:pointer; width: auto; font-size:11px !important; line-height:13px; padding:3px 11px; margin-top: 12px; text-decoration:none; -moz-border-radius:11px; border-radius:11px; -webkit-border-radius:11px } + .button-primary { background-color:#21759B; font-weight: bold; border-color:#298CBA; text-align:center; color:#EAF2FA; text-shadow:0 -1px 0 rgba(0, 0, 0, 0.3); border-style:solid; border-width:1px; cursor:pointer; width: auto; font-size:11px !important; line-height:13px; padding:3px 11px; margin-top: 21px; text-decoration:none; -moz-border-radius:11px; border-radius:11px; -webkit-border-radius:11px } + .clear { clear: both; } + .fb-add-field { padding-left: 10px; } + .fb-add-option { margin: 0 0 14px 100px; } + .fb-container { margin: 21px; padding-bottom: 20px; } + .fb-desc, #fb-add-field { margin-top: 34px; } + .fb-extra-fields { margin-bottom: 2px; } + .fb-form-case { background: #FFF; padding: 13px; border: 1px solid #E2E2E2; width: 336px; -moz-border-radius:4px; border-radius:4px; -webkit-border-radius:4px } + .fb-form-case a { outline: none; } + .fb-form-case input[type='text'], .fb-form-case textarea { background: #E1E1E1; } + .fb-radio-label { display: inline-block; float: left; width: 290px; } + .fb-new-fields { position: relative; border: 1px dashed #FFF; background: #FFF; padding: 4px 10px 10px; cursor: default; } + .fb-new-fields:hover { border: 1px dashed #BBDBEA; background: #F7FBFD; } + .fb-options { width: 170px !important; } + .fb-remove { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-field.gif') no-repeat; position: absolute; cursor: pointer !important; right: -26px; top: 27px; width: 20px; height: 23px; } + .fb-remove:hover { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-field-hover.gif') no-repeat; } + .fb-remove-small { top: 2px !important; } + .fb-remove-option { position: absolute; top: 1px; right: 10px; width: 20px; height: 23px; background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-option.gif') no-repeat; } + .fb-remove-option:hover { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-option-hover.gif') no-repeat; } + .fb-reorder { cursor: move; position: relative; } + .fb-reorder:hover div { display: block !important; width: 130px !important; position: absolute; top: 0; right: 0; z-index: 200; padding: 5px 10px; color: #555; font-size: 11px; background: #FFF; border: 1px solid #CCC; -moz-border-radius:4px; border-radius:4px; -webkit-border-radius:4px; } + .fb-right { position: absolute; right: 0; top: 0; width: 315px; margin: 57px 21px 0 0; } + .fb-right .fb-new-fields { border: none; background: #F9F9F9; padding: 0; } + .fb-right input[type='text'] { width: 195px; margin-bottom: 14px; } + .fb-right label { color: #444; width: 100px; float: left; font-weight: normal; } + .fb-right select { width: 195px !important; margin-bottom: 14px; } + .fb-right textarea { margin-bottom: 13px; } + .fb-right p { color: #999; line-height: 19px; } + .fb-settings input[type='text'], .fb-settings textarea { background-image: none !important; } + .fb-success { position: absolute; top: -3px; right: 100px; padding: 6px 23px 4px 23px; background: #FFFFE0; font-weight: normal; border: 1px solid #E6DB55; color: #333; -moz-border-radius:4px; border-radius:4px; -webkit-border-radius:4px; } + .right { float: right; } + /* rtl */ + body.rtl{ direction: rtl; font-family:Tahoma,Arial,sans-serif} + .rtl input[type='text'] { margin-left: 4px; margin-right: 0; } + .rtl input[type='checkbox'], .rtl input[type='radio'] { float: right; } + .rtl input[type='radio'] { margin-left: 8px; margin-right: 0; } + .rtl label.radio { margin: -2px 5px 0 0; } + .rtl label span.label-required { margin-right: 4px; margin-left:0 } + .rtl #sidemenu { padding-right:10px; padding-left: 0; left:auto; right: 0; } + .rtl #sidemenu a { float:right; } + .rtl .fb-add-field { padding-right: 10px; padding-left: 0; } + .rtl .fb-add-option { margin: 0 100px 14px 0; } + .rtl .fb-radio-label { margin-right: 8px; margin-left: 0; float: right; } + .rtl .fb-remove { right: auto; left: -26px; transform: scaleX(-1); } + .rtl .fb-remove-option { right: auto; left: 10px; } + .rtl .fb-reorder:hover div { left: 0; right: auto; } + .rtl .fb-right { left: 0; right: auto; margin: 57px 0 0 21px; } + .rtl .fb-right label { float: right; } + .rtl .fb-success { right: auto; left: 100px;} + .rtl .right { float: left; } + @media only screen and (min--moz-device-pixel-ratio: 1.5), only screen and (-o-min-device-pixel-ratio: 3/2), only screen and (-webkit-min-device-pixel-ratio: 1.5), only screen and (min-device-pixel-ratio: 1.5) { + .fb-remove { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-field-2x.png') no-repeat; background-size: 20px 23px; } + .fb-remove:hover { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-field-hover-2x.png') no-repeat; background-size: 20px 23px; } + .fb-remove-option { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-option-2x.png') no-repeat; background-size: 20px 23px; } + .fb-remove-option:hover { background: url('<?php echo GRUNION_PLUGIN_URL; ?>/images/grunion-remove-option-hover-2x.png') no-repeat; background-size: 20px 23px; } + } +</style> +</head> + +<body +<?php +if ( is_rtl() ) { + echo 'class="rtl"'; } +?> +> + <div id="media-upload-header"> + <div id="fb-success" class="fb-success" style="display: none;"><?php esc_html_e( 'Your new field was saved successfully', 'jetpack' ); ?></div> + <ul id="sidemenu"> + <li id="tab-preview"><a class="current" href=""><?php esc_html_e( 'Form builder', 'jetpack' ); ?></a></li> + <li id="tab-settings"><a href=""><?php esc_html_e( 'Email notifications', 'jetpack' ); ?></a></li> + </ul> + </div> + <div class="fb-right"> + <div id="fb-desc" class="fb-desc"> + <h3><?php esc_html_e( 'How does this work?', 'jetpack' ); ?></h3> + <p><?php esc_html_e( 'By adding a contact form, your readers will be able to submit feedback to you. All feedback is automatically scanned for spam, and the legitimate feedback will be emailed to you.', 'jetpack' ); ?></p> + <h3 style="margin-top: 21px;"><?php esc_html_e( 'Can I add more fields?', 'jetpack' ); ?></h3> + <p> + <?php + printf( + esc_html( _x( 'Sure thing. %1$s to add a new text box, textarea, radio, checkbox, or dropdown field.', '%1$s = "Click here" in an HTML link', 'jetpack' ) ), + '<a href="#" class="fb-add-field" style="padding-left: 0;">' . esc_html__( 'Click here', 'jetpack' ) . '</a>' + ); + ?> + </p> + <h3 style="margin-top: 21px;"><?php esc_html_e( 'Can I view my feedback within WordPress?', 'jetpack' ); ?></h3> + <p> + <?php + printf( + esc_html( _x( 'Yep, you can read your feedback at any time by clicking the "%1$s" link in the admin menu.', '%1$s = "Feedback" in an HTML link', 'jetpack' ) ), + '<a id="fb-feedback" href="' . admin_url( 'edit.php?post_type=feedback' ) . '">' . esc_html__( 'Feedback', 'jetpack' ) . '</a>' + ); + ?> + </p> + <div class="clear"></div> + </div> + <div id="fb-email-desc" class="fb-desc" style="display: none;"> + <h3><?php esc_html_e( 'Do I need to fill this out?', 'jetpack' ); ?></h3> + <p><?php esc_html_e( 'Nope. However, if you’d like to modify where your feedback is sent, or the subject line you can. If you don’t make any changes here, feedback will be sent to the author of the page/post and the subject will be the name of this page/post.', 'jetpack' ); ?></p> + <h3 style="margin-top: 21px;"><?php esc_html_e( 'Can I send a notification to more than one person?', 'jetpack' ); ?></h3> + <p><?php esc_html_e( 'Yep. You can enter multiple email addresses in the Email address field, and separate them with commas. A notification email will then be sent to each email address.', 'jetpack' ); ?></p> + <div class="clear"></div> + </div> + <div id="fb-add-field" style="display: none;"> + <h3><?php esc_html_e( 'Edit this new field', 'jetpack' ); ?></h3> + + <label for="fb-new-label"><?php esc_html_e( 'Label', 'jetpack' ); ?></label> + <input type="text" id="fb-new-label" value="<?php esc_attr_e( 'New field', 'jetpack' ); ?>" /> + + <label for="fb-new-label"><?php esc_html_e( 'Field type', 'jetpack' ); ?></label> + <select id="fb-new-type"> + <option value="checkbox"><?php esc_html_e( 'Checkbox', 'jetpack' ); ?></option> + <option value="checkbox-multiple"><?php esc_html_e( 'Checkbox with Multiple Items', 'jetpack' ); ?></option> + <option value="select"><?php esc_html_e( 'Drop down', 'jetpack' ); ?></option> + <option value="email"><?php esc_html_e( 'Email', 'jetpack' ); ?></option> + <option value="name"><?php esc_html_e( 'Name', 'jetpack' ); ?></option> + <option value="radio"><?php esc_html_e( 'Radio', 'jetpack' ); ?></option> + <option value="text" selected="selected"><?php esc_html_e( 'Text', 'jetpack' ); ?></option> + <option value="textarea"><?php esc_html_e( 'Textarea', 'jetpack' ); ?></option> + <option value="url"><?php esc_html_e( 'Website', 'jetpack' ); ?></option> + </select> + <div class="clear"></div> + + <div id="fb-options" style="display: none;"> + <div id="fb-new-options"> + <label for="fb-option0"><?php esc_html_e( 'Options', 'jetpack' ); ?></label> + <input type="text" id="fb-option0" optionid="0" value="<?php esc_attr_e( 'First option', 'jetpack' ); ?>" class="fb-options" /> + </div> + <div id="fb-add-option" class="fb-add-option"> + <a href="#" id="fb-another-option"><?php esc_html_e( 'Add another option', 'jetpack' ); ?></a> + </div> + </div> + + <div class="fb-required"> + <label for="fb-new-label"></label> + <input type="checkbox" id="fb-new-required" /> + <label for="fb-new-label" class="fb-radio-label"><?php esc_html_e( 'Required?', 'jetpack' ); ?></label> + <div class="clear"></div> + </div> + + <input type="hidden" id="fb-field-id" /> + <input type="submit" class="button" value="<?php esc_attr_e( 'Save this field', 'jetpack' ); ?>" id="fb-save-field" name="save"> + </div> + </div> + <form id="fb-preview"> + <div id="fb-preview-form" class="fb-container"> + <h1><?php esc_html_e( 'Here’s what your form will look like', 'jetpack' ); ?></h1> + <div id="sortable" class="fb-form-case"> + + <div id="fb-extra-fields" class="fb-extra-fields"></div> + + <a href="#" id="fb-new-field" class="fb-add-field"><?php esc_html_e( 'Add a new field', 'jetpack' ); ?></a> + </div> + <input type="submit" class="button-primary" tabindex="4" value="<?php esc_attr_e( 'Add this form to my post', 'jetpack' ); ?>" id="fb-save-form" name="save"> + </div> + <div id="fb-email-settings" class="fb-container" style="display: none;"> + <h1><?php esc_html_e( 'Email settings', 'jetpack' ); ?></h1> + <div class="fb-form-case fb-settings"> + <label for="fb-fieldname"><?php esc_html_e( 'Enter your email address', 'jetpack' ); ?></label> + <input type="text" id="fb-field-my-email" style="background: #FFF !important;" /> + + <label for="fb-fieldemail" style="margin-top: 14px;"><?php esc_html_e( 'What should the subject line be?', 'jetpack' ); ?></label> + <input type="text" id="fb-field-subject" style="background: #FFF !important;" /> + </div> + <input type="submit" class="button-primary" value="<?php esc_attr_e( 'Save and go back to form builder', 'jetpack' ); ?>" id="fb-prev-form" name="save"> + </div> + </form> +</body> +</html> diff --git a/plugins/jetpack/modules/contact-form/images/blank-screen-akismet.png b/plugins/jetpack/modules/contact-form/images/blank-screen-akismet.png Binary files differnew file mode 100644 index 00000000..a4ba1e2d --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/blank-screen-akismet.png diff --git a/plugins/jetpack/modules/contact-form/images/blank-screen-button.png b/plugins/jetpack/modules/contact-form/images/blank-screen-button.png Binary files differnew file mode 100644 index 00000000..58dfa26b --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/blank-screen-button.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-form-2x.png b/plugins/jetpack/modules/contact-form/images/grunion-form-2x.png Binary files differnew file mode 100644 index 00000000..824d85a3 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-form-2x.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-form.png b/plugins/jetpack/modules/contact-form/images/grunion-form.png Binary files differnew file mode 100644 index 00000000..f4a0cc1d --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-form.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-field-2x.png b/plugins/jetpack/modules/contact-form/images/grunion-remove-field-2x.png Binary files differnew file mode 100644 index 00000000..bfbca5ed --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-field-2x.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-field-hover-2x.png b/plugins/jetpack/modules/contact-form/images/grunion-remove-field-hover-2x.png Binary files differnew file mode 100644 index 00000000..dca39da9 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-field-hover-2x.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-field-hover.gif b/plugins/jetpack/modules/contact-form/images/grunion-remove-field-hover.gif Binary files differnew file mode 100644 index 00000000..20d9e712 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-field-hover.gif diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-field.gif b/plugins/jetpack/modules/contact-form/images/grunion-remove-field.gif Binary files differnew file mode 100644 index 00000000..55062664 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-field.gif diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-option-2x.png b/plugins/jetpack/modules/contact-form/images/grunion-remove-option-2x.png Binary files differnew file mode 100644 index 00000000..4272442c --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-option-2x.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-option-hover-2x.png b/plugins/jetpack/modules/contact-form/images/grunion-remove-option-hover-2x.png Binary files differnew file mode 100644 index 00000000..210de1b3 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-option-hover-2x.png diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-option-hover.gif b/plugins/jetpack/modules/contact-form/images/grunion-remove-option-hover.gif Binary files differnew file mode 100644 index 00000000..9098b065 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-option-hover.gif diff --git a/plugins/jetpack/modules/contact-form/images/grunion-remove-option.gif b/plugins/jetpack/modules/contact-form/images/grunion-remove-option.gif Binary files differnew file mode 100644 index 00000000..ec491663 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/images/grunion-remove-option.gif diff --git a/plugins/jetpack/modules/contact-form/js/editor-view.js b/plugins/jetpack/modules/contact-form/js/editor-view.js new file mode 100644 index 00000000..b62d92f8 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/js/editor-view.js @@ -0,0 +1,284 @@ +/* global grunionEditorView, tinyMCE, QTags, wp */ +( function( $, wp, grunionEditorView ) { + wp.mce = wp.mce || {}; + if ( 'undefined' === typeof wp.mce.views ) { + return; + } + + wp.mce.grunion_wp_view_renderer = { + shortcode_string: 'contact-form', + template: wp.template( 'grunion-contact-form' ), + field_templates: { + email: wp.template( 'grunion-field-email' ), + telephone: wp.template( 'grunion-field-telephone' ), + textarea: wp.template( 'grunion-field-textarea' ), + radio: wp.template( 'grunion-field-radio' ), + checkbox: wp.template( 'grunion-field-checkbox' ), + 'checkbox-multiple': wp.template( 'grunion-field-checkbox-multiple' ), + select: wp.template( 'grunion-field-select' ), + date: wp.template( 'grunion-field-date' ), + text: wp.template( 'grunion-field-text' ), + name: wp.template( 'grunion-field-text' ), + url: wp.template( 'grunion-field-url' ), + }, + edit_template: wp.template( 'grunion-field-edit' ), + editor_inline: wp.template( 'grunion-editor-inline' ), + editor_option: wp.template( 'grunion-field-edit-option' ), + getContent: function() { + var content = this.shortcode.content, + index = 0, + field, + named, + body = ''; + + // If it's the legacy `[contact-form /]` syntax, populate default fields. + if ( ! content ) { + content = grunionEditorView.default_form; + } + + // Render the fields. + while ( ( field = wp.shortcode.next( 'contact-field', content, index ) ) ) { + index = field.index + field.content.length; + named = field.shortcode.attrs.named; + if ( ! named.type || ! this.field_templates[ named.type ] ) { + named.type = 'text'; + } + if ( named.required ) { + named.required = grunionEditorView.labels.required_field_text; + } + if ( named.options && 'string' === typeof named.options ) { + named.options = named.options.split( ',' ); + } + body += this.field_templates[ named.type ]( named ); + } + + var options = { + body: body, + submit_button_text: grunionEditorView.labels.submit_button_text, + }; + + return this.template( options ); + }, + edit: function( data, update_callback ) { + var shortcode_data = wp.shortcode.next( this.shortcode_string, data ), + shortcode = shortcode_data.shortcode, + $tinyMCE_document = $( tinyMCE.activeEditor.getDoc() ), + $view = $tinyMCE_document.find( '.wpview.wpview-wrap' ).filter( function() { + return $( this ).attr( 'data-mce-selected' ); + } ), + $editframe = $( '<iframe scrolling="no" class="inline-edit-contact-form" />' ), + index = 0, + named, + fields = '', + field; + + if ( ! shortcode.content ) { + shortcode.content = grunionEditorView.default_form; + } + + // Render the fields. + while ( ( field = wp.shortcode.next( 'contact-field', shortcode.content, index ) ) ) { + index = field.index + field.content.length; + named = field.shortcode.attrs.named; + if ( named.options && 'string' === typeof named.options ) { + named.options = named.options.split( ',' ); + } + fields += this.edit_template( named ); + } + + $editframe.on( 'checkheight', function() { + var innerDoc = this.contentDocument ? this.contentDocument : this.contentWindow.document; + this.style.height = '10px'; + this.style.height = 5 + innerDoc.body.scrollHeight + 'px'; + tinyMCE.activeEditor.execCommand( 'wpAutoResize' ); + } ); + + $editframe.on( 'load', function() { + var stylesheet_url = + 1 === window.isRtl + ? grunionEditorView.inline_editing_style_rtl + : grunionEditorView.inline_editing_style, + $stylesheet = $( '<link rel="stylesheet" href="' + stylesheet_url + '" />' ), + $dashicons_css = $( + '<link rel="stylesheet" href="' + grunionEditorView.dashicons_css_url + '" />' + ); + + $stylesheet.on( 'load', function() { + $editframe + .contents() + .find( 'body' ) + .css( 'visibility', 'visible' ); + $editframe.trigger( 'checkheight' ); + } ); + $editframe + .contents() + .find( 'head' ) + .append( $stylesheet ) + .append( $dashicons_css ); + + $editframe + .contents() + .find( 'body' ) + .html( + wp.mce.grunion_wp_view_renderer.editor_inline( { + to: shortcode.attrs.named.to, + subject: shortcode.attrs.named.subject, + fields: fields, + } ) + ) + .css( 'visibility', 'hidden' ); + + $editframe + .contents() + .find( 'input:first' ) + .focus(); + + setTimeout( function() { + $editframe.trigger( 'checkheight' ); + }, 250 ); + + // Add a second timeout for super long forms racing, and to not slow it down for shorter forms unnecessarily. + setTimeout( function() { + $editframe.trigger( 'checkheight' ); + }, 500 ); + + var $editfields = $editframe.contents().find( '.grunion-fields' ), + $buttons = $editframe.contents().find( '.grunion-controls' ); + + $editfields.sortable(); + + // Now, add all the listeners! + + $editfields.on( 'change select', 'select[name=type]', function() { + $( this ).closest( '.grunion-field-edit' )[ 0 ].className = + 'card is-compact grunion-field-edit grunion-field-' + $( this ).val(); + $editframe.trigger( 'checkheight' ); + } ); + + $editfields.on( 'click', '.delete-option', function( e ) { + e.preventDefault(); + $( this ) + .closest( 'li' ) + .remove(); + $editframe.trigger( 'checkheight' ); + } ); + + $editfields.on( 'click', '.add-option', function( e ) { + var $new_option = $( wp.mce.grunion_wp_view_renderer.editor_option() ); + e.preventDefault(); + $( this ) + .closest( 'li' ) + .before( $new_option ); + $editframe.trigger( 'checkheight' ); + $new_option.find( 'input:first' ).focus(); + } ); + + $editfields.on( 'click', '.delete-field', function( e ) { + e.preventDefault(); + $( this ) + .closest( '.card' ) + .remove(); + $editframe.trigger( 'checkheight' ); + } ); + + $buttons.find( 'input[name=submit]' ).on( 'click', function() { + var new_data = shortcode; + + new_data.type = 'closed'; + new_data.attrs = {}; + new_data.content = ''; + + $editfields.children().each( function() { + var field_shortcode = { + tag: 'contact-field', + type: 'single', + attrs: { + label: $( this ) + .find( 'input[name=label]' ) + .val(), + type: $( this ) + .find( 'select[name=type]' ) + .val(), + }, + }, + options = []; + + if ( $( this ).find( 'input[name=required]:checked' ).length ) { + field_shortcode.attrs.required = '1'; + } + + $( this ) + .find( 'input[name=option]' ) + .each( function() { + if ( $( this ).val() ) { + options.push( $( this ).val() ); + } + } ); + if ( options.length ) { + field_shortcode.attrs.options = options.join( ',' ); + } + + new_data.content += wp.shortcode.string( field_shortcode ); + } ); + + if ( + $editframe + .contents() + .find( 'input[name=to]' ) + .val() + ) { + new_data.attrs.to = $editframe + .contents() + .find( 'input[name=to]' ) + .val(); + } + if ( + $editframe + .contents() + .find( 'input[name=subject]' ) + .val() + ) { + new_data.attrs.subject = $editframe + .contents() + .find( 'input[name=subject]' ) + .val(); + } + + update_callback( wp.shortcode.string( new_data ) ); + } ); + + $buttons.find( 'input[name=cancel]' ).on( 'click', function() { + update_callback( wp.shortcode.string( shortcode ) ); + } ); + + $buttons.find( 'input[name=add-field]' ).on( 'click', function() { + var $new_field = $( wp.mce.grunion_wp_view_renderer.edit_template( {} ) ); + $editfields.append( $new_field ); + $editfields.sortable( 'refresh' ); + $editframe.trigger( 'checkheight' ); + $new_field.find( 'input:first' ).focus(); + } ); + } ); + + $view.html( $editframe ); + }, + }; + wp.mce.views.register( 'contact-form', wp.mce.grunion_wp_view_renderer ); + + // Add the 'text' editor button. + QTags.addButton( 'grunion_shortcode', grunionEditorView.labels.quicktags_label, function() { + QTags.insertContent( '[contact-form]' + grunionEditorView.default_form + '[/contact-form]' ); + } ); + + var $wp_content_wrap = $( '#wp-content-wrap' ); + $( '#insert-jetpack-contact-form' ).on( 'click', function( e ) { + e.preventDefault(); + if ( $wp_content_wrap.hasClass( 'tmce-active' ) ) { + tinyMCE.execCommand( 'grunion_add_form' ); + } else if ( $wp_content_wrap.hasClass( 'html-active' ) ) { + QTags.insertContent( '[contact-form]' + grunionEditorView.default_form + '[/contact-form]' ); + } else { + window.console.error( 'Neither TinyMCE nor QuickTags is active. Unable to insert form.' ); + } + } ); +} )( jQuery, wp, grunionEditorView ); diff --git a/plugins/jetpack/modules/contact-form/js/grunion-admin.js b/plugins/jetpack/modules/contact-form/js/grunion-admin.js new file mode 100644 index 00000000..100fb22b --- /dev/null +++ b/plugins/jetpack/modules/contact-form/js/grunion-admin.js @@ -0,0 +1,30 @@ +/* global ajaxurl */ +jQuery( function( $ ) { + $( document ).on( 'click', '#jetpack-check-feedback-spam:not(.button-disabled)', function( e ) { + e.preventDefault(); + + $( '#jetpack-check-feedback-spam:not(.button-disabled)' ).addClass( 'button-disabled' ); + $( '.jetpack-check-feedback-spam-spinner' ) + .addClass( 'spinner' ) + .show(); + grunion_check_for_spam( 0, 100 ); + } ); + + function grunion_check_for_spam( offset, limit ) { + $.post( + ajaxurl, + { + action: 'grunion_recheck_queue', + offset: offset, + limit: limit, + }, + function( result ) { + if ( result.processed < limit ) { + window.location.reload(); + } else { + grunion_check_for_spam( offset + limit, limit ); + } + } + ); + } +} ); diff --git a/plugins/jetpack/modules/contact-form/js/grunion-frontend.js b/plugins/jetpack/modules/contact-form/js/grunion-frontend.js new file mode 100644 index 00000000..1c1819a5 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/js/grunion-frontend.js @@ -0,0 +1,3 @@ +jQuery( function( $ ) { + $( '.contact-form input.jp-contact-form-date' ).datepicker(); +} ); diff --git a/plugins/jetpack/modules/contact-form/js/grunion.js b/plugins/jetpack/modules/contact-form/js/grunion.js new file mode 100644 index 00000000..e34fc135 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/js/grunion.js @@ -0,0 +1,1180 @@ +/* jshint onevar: false, devel: true, smarttabs: true */ +/* global GrunionFB_i18n: true, FB, ajax_nonce_shortcode, ajax_nonce_json, ajaxurl, postId */ + +if ( ! window.FB ) { + window.FB = {}; +} + +GrunionFB_i18n = jQuery.extend( + { + nameLabel: 'Name', + emailLabel: 'Email', + urlLabel: 'Website', + commentLabel: 'Comment', + newLabel: 'New Field', + optionsLabel: 'Options', + optionLabel: 'Option', + firstOptionLabel: 'First option', + problemGeneratingForm: + "Oops, there was a problem generating your form. You'll likely need to try again.", + moveInstructions: 'Drag up or down\nto re-arrange', + moveLabel: 'move', + editLabel: 'edit', + savedMessage: 'Saved successfully', + requiredLabel: '(required)', + exitConfirmMessage: + 'Are you sure you want to exit the form editor without saving? Any changes you have made will be lost.', + maxNewFields: 5, + invalidEmail: ' is an invalid email address.', + }, + GrunionFB_i18n +); + +GrunionFB_i18n.moveInstructions = GrunionFB_i18n.moveInstructions.replace( '\n', '<br />' ); + +FB.span = jQuery( '<span>' ); +FB.esc_html = function( string ) { + return FB.span.text( string ).html(); +}; + +FB.esc_attr = function( string ) { + string = FB.esc_html( string ); + return string.replace( '"', '"' ).replace( "'", ''' ); +}; + +FB.ContactForm = ( function() { + var fbForm = { + // Main object that generated shortcode via AJAX call + action: 'grunion_shortcode', + _ajax_nonce: ajax_nonce_shortcode, + to: '', + subject: '', + fields: {}, + }; + var defaultFields = { + name: { + label: GrunionFB_i18n.nameLabel, + type: 'name', + required: true, + options: [], + order: '1', + }, + email: { + label: GrunionFB_i18n.emailLabel, + type: 'email', + required: true, + options: [], + order: '2', + }, + url: { + label: GrunionFB_i18n.urlLabel, + type: 'url', + required: false, + options: [], + order: '3', + }, + comment: { + label: GrunionFB_i18n.commentLabel, + type: 'textarea', + required: true, + options: [], + order: '4', + }, + }; + var debug = false; // will print errors to log if true + var grunionNewCount = 0; // increment for new fields + var maxNewFields = GrunionFB_i18n.maxNewFields; // See filter in ../grunion-form-view.php + var optionsCache = {}; + var optionsCount = 0; // increment for options + var shortcode; + + function addField() { + try { + grunionNewCount++; + if ( grunionNewCount <= maxNewFields ) { + // Add to preview + jQuery( '#fb-extra-fields' ).append( + '<div id="fb-new-field' + + grunionNewCount + + '" fieldid="' + + grunionNewCount + + '" class="fb-new-fields"><div class="fb-fields"><div id="' + + grunionNewCount + + '" class="fb-remove"></div><label fieldid="' + + grunionNewCount + + '" for="fb-field' + + grunionNewCount + + '"><span class="label-text">' + + GrunionFB_i18n.newLabel + + '</span> </label><input type="text" id="fb-field' + + grunionNewCount + + '" disabled="disabled" /></div></div>' + ); + // Add to form object + fbForm.fields[ grunionNewCount ] = { + label: GrunionFB_i18n.newLabel, + type: 'text', + required: false, + options: [], + order: '5', + }; + if ( grunionNewCount === maxNewFields ) { + jQuery( '#fb-new-field' ).hide(); + } + // Reset form for this new field + optionsCount = 0; + optionsCache = {}; + jQuery( '#fb-new-options' ).html( + '<label for="fb-option0">' + + GrunionFB_i18n.optionsLabel + + '</label><input type="text" id="fb-option0" optionid="0" value="' + + GrunionFB_i18n.firstOptionLabel + + '" class="fb-options" />' + ); + jQuery( '#fb-options' ).hide(); + jQuery( '#fb-new-label' ).val( GrunionFB_i18n.newLabel ); + jQuery( '#fb-new-type' ).val( 'text' ); + jQuery( '#fb-field-id' ).val( grunionNewCount ); + setTimeout( function() { + jQuery( '#fb-new-label' ) + .focus() + .select(); + }, 100 ); + } else { + jQuery( '#fb-new-field' ).hide(); + } + } catch ( e ) { + if ( debug ) { + console.log( 'addField(): ' + e ); + } + } + } + function addOption() { + try { + optionsCount = jQuery( '#fb-new-options .fb-options' ).length; + var thisId = jQuery( '#fb-field-id' ).val(); + var thisType = jQuery( '#fb-new-type' ).val(); + if ( thisType === 'radio' ) { + // Add to right col + jQuery( '#fb-new-options' ).append( + '<div id="fb-option-box-' + + optionsCount + + '" class="fb-new-fields"><span optionid="' + + optionsCount + + '" class="fb-remove-option"></span><label></label><input type="text" id="fb-option' + + optionsCount + + '" optionid="' + + optionsCount + + '" value="' + + GrunionFB_i18n.optionLabel + + '" class="fb-options" /><div>' + ); + // Add to preview + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).append( + '<div id="fb-radio-' + + thisId + + '-' + + optionsCount + + '"><input type="radio" disabled="disabled" id="fb-field' + + thisId + + '" name="radio-' + + thisId + + '" /><span>' + + GrunionFB_i18n.optionLabel + + '</span><div class="clear"></div></div>' + ); + } else if ( 'checkbox-multiple' === thisType ) { + // Add to right col + jQuery( '#fb-new-options' ).append( + '<div id="fb-option-box-' + + optionsCount + + '" class="fb-new-fields"><span optionid="' + + optionsCount + + '" class="fb-remove-option"></span><label></label><input type="text" id="fb-option' + + optionsCount + + '" optionid="' + + optionsCount + + '" value="' + + GrunionFB_i18n.optionLabel + + '" class="fb-options" /><div>' + ); + // Add to preview + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).append( + '<div id="fb-checkbox-multiple-' + + thisId + + '-' + + optionsCount + + '"><input type="checkbox" disabled="disabled" id="fb-field' + + thisId + + '" name="checkbox-multiple-' + + thisId + + '" /><span>' + + GrunionFB_i18n.optionLabel + + '</span><div class="clear"></div></div>' + ); + } else { + // Add to right col + jQuery( '#fb-new-options' ).append( + '<div id="fb-option-box-' + + optionsCount + + '" class="fb-new-fields"><span optionid="' + + optionsCount + + '" class="fb-remove-option"></span><label></label><input type="text" id="fb-option' + + optionsCount + + '" optionid="' + + optionsCount + + '" value="" class="fb-options" /><div>' + ); + // Add to preview + jQuery( '#fb-field' + thisId ).append( + '<option id="fb-' + + thisId + + '-' + + optionsCount + + '" value="' + + thisId + + '-' + + optionsCount + + '"></option>' + ); + } + // Add to fbForm object + fbForm.fields[ thisId ].options[ optionsCount ] = ''; + // Add focus to new field + jQuery( '#fb-option' + optionsCount ) + .focus() + .select(); + } catch ( e ) { + if ( debug ) { + console.log( 'addOption(): ' + e ); + } + } + } + function buildPreview() { + try { + if ( fbForm.to ) { + jQuery( '#fb-field-my-email' ).val( fbForm.to ); + } + if ( fbForm.subject ) { + jQuery( '#fb-field-subject' ).val( fbForm.subject ); + } + // Loop over and add fields + jQuery.each( fbForm.fields, function( index, value ) { + jQuery( '#fb-extra-fields' ).before( + '<div class="fb-new-fields ui-state-default" fieldid="' + + index + + '" id="fb-new-field' + + index + + '"><div class="fb-fields"></div></div>' + ); + jQuery( '#fb-field-id' ).val( index ); + optionsCache[ index ] = {}; + optionsCache[ index ].options = []; + if ( + 'radio' === value.type || + 'select' === value.type || + 'checkbox-multiple' === value.type + ) { + jQuery.each( value.options, function( i, value ) { + optionsCache[ index ].options[ i ] = value; + } ); + } + updateType( value.type, value.label, value.required ); + } ); + } catch ( e ) { + if ( debug ) { + console.log( 'buildPreview(): ' + e ); + } + } + } + function customOptions( id, thisType ) { + try { + var thisOptions = ''; + for ( var i = 0; i < optionsCache[ id ].options.length; i++ ) { + if ( optionsCache[ id ].options[ i ] !== undefined ) { + if ( thisType === 'radio' ) { + thisOptions = + thisOptions + + '<div id="fb-radio-' + + id + + '-' + + i + + '"><input type="radio" id="fb-field' + + id + + '" name="radio-' + + id + + '" /><span>' + + FB.esc_html( optionsCache[ id ].options[ i ] ) + + '</span><div class="clear"></div></div>'; + } else if ( 'checkbox-multiple' === thisType ) { + thisOptions = + thisOptions + + '<div id="fb-checkbox-multiple-' + + id + + '-' + + i + + '"><input type="checkbox" id="fb-field' + + id + + '" name="checkbox-multiple-' + + id + + '" /><span>' + + FB.esc_html( optionsCache[ id ].options[ i ] ) + + '</span><div class="clear"></div></div>'; + } else { + thisOptions = + thisOptions + + '<option id="fb-' + + id + + '-' + + i + + '" value="' + + id + + '-' + + i + + '">' + + FB.esc_html( optionsCache[ id ].options[ i ] ) + + '</option>'; + } + } + } + return thisOptions; + } catch ( e ) { + if ( debug ) { + console.log( 'customOptions(): ' + e ); + } + } + } + function deleteField( that ) { + try { + grunionNewCount--; + var thisId = that.attr( 'id' ); + delete fbForm.fields[ thisId ]; + jQuery( '#' + thisId ) + .parent() + .parent() + .remove(); + if ( grunionNewCount <= maxNewFields ) { + jQuery( '#fb-new-field' ).show(); + } + } catch ( e ) { + if ( debug ) { + console.log( 'deleteField(): ' + e ); + } + } + } + function editField( that ) { + try { + scroll( 0, 0 ); + setTimeout( function() { + jQuery( '#fb-new-label' ) + .focus() + .select(); + }, 100 ); + var thisId = that.parent().attr( 'fieldid' ); + loadFieldEditor( thisId ); + } catch ( e ) { + if ( debug ) { + console.log( 'editField(): ' + e ); + } + } + } + function grabShortcode() { + try { + // Takes fbForm object and returns shortcode syntax + jQuery.post( ajaxurl, fbForm, function( response ) { + shortcode = response; + } ); + } catch ( e ) { + alert( GrunionFB_i18n.problemGeneratingForm ); + if ( debug ) { + console.log( 'grabShortcode(): ' + e ); + } + } + } + function hideDesc() { + jQuery( '#fb-desc' ).hide(); + jQuery( '#fb-add-field' ).show(); + } + function hidePopup() { + try { + // copied from wp-includes/js/thickbox/thickbox.js + jQuery( '#TB_imageOff', window.parent.document ).unbind( 'click' ); + jQuery( '#TB_closeWindowButton', window.parent.document ).unbind( 'click' ); + jQuery( '#TB_window', window.parent.document ).fadeOut( 'fast' ); + jQuery( 'body', window.parent.document ).removeClass( 'modal-open' ); + jQuery( '#TB_window,#TB_overlay,#TB_HideSelect', window.parent.document ) + .trigger( 'unload' ) + .unbind() + .remove(); + jQuery( '#TB_load', window.parent.document ).remove(); + if ( typeof window.parent.document.body.style.maxHeight === 'undefined' ) { + //if IE 6 + jQuery( 'body', 'html', window.parent.document ).css( { height: 'auto', width: 'auto' } ); + jQuery( 'html', window.parent.document ).css( 'overflow', '' ); + } + window.parent.document.onkeydown = ''; + window.parent.document.onkeyup = ''; + return false; + } catch ( e ) { + if ( debug ) { + console.log( 'hidePopup(): ' + e ); + } + } + } + function hideShowEditLink( whichType, that ) { + try { + if ( whichType === 'show' ) { + // Prevents showing links twice + if ( jQuery( '.fb-edit-field' ).is( ':visible' ) ) { + jQuery( '.fb-edit-field' ).remove(); + } + that + .find( 'label' ) + .prepend( + '<span class="right fb-edit-field" style="font-weight: normal;"><a href="" class="fb-reorder"><div style="display: none;">' + + GrunionFB_i18n.moveInstructions + + '</div>' + + GrunionFB_i18n.moveLabel + + '</a> <span style="color: #C7D8DE;">|</span> <a href="" class="fb-edit">' + + GrunionFB_i18n.editLabel + + '</a></span>' + ); + } else { + jQuery( '.fb-edit-field' ).remove(); + } + } catch ( e ) { + if ( debug ) { + console.log( 'hideShowEditLink(): ' + e ); + } + } + } + function loadFieldEditor( id ) { + try { + var thisType = fbForm.fields[ id ].type; + jQuery( '#fb-options' ).hide(); + // Reset hidden field ID + jQuery( '#fb-field-id' ).val( id ); + // Load label + jQuery( '#fb-new-label' ).val( fbForm.fields[ id ].label ); + // Load type + jQuery( '#fb-new-type' ).val( fbForm.fields[ id ].type ); + // Load required + if ( fbForm.fields[ id ].required ) { + jQuery( '#fb-new-required' ).prop( 'checked', true ); + } else { + jQuery( '#fb-new-required' ).prop( 'checked', false ); + } + // Load options if there are any + if ( 'select' === thisType || 'radio' === thisType || 'checkbox-multiple' === thisType ) { + var thisOptions = fbForm.fields[ id ].options; + jQuery( '#fb-options' ).show(); + jQuery( '#fb-new-options' ).html( '' ); // Clear it all out + for ( var i = 0; i < thisOptions.length; i++ ) { + if ( thisOptions[ i ] !== undefined ) { + if ( thisType === 'radio' ) { + jQuery( '#fb-new-options' ).append( + '<div id="fb-option-box-' + + i + + '" class="fb-new-fields"><span optionid="' + + i + + '" class="fb-remove-option"></span><label></label><input type="text" id="fb-option' + + i + + '" optionid="' + + i + + '" value="' + + FB.esc_attr( fbForm.fields[ id ].options[ i ] ) + + '" class="fb-options" /><div>' + ); + } else { + jQuery( '#fb-new-options' ).append( + '<div id="fb-option-box-' + + i + + '" class="fb-new-fields"><span optionid="' + + i + + '" class="fb-remove-option"></span><label></label><input type="text" id="fb-option' + + i + + '" optionid="' + + i + + '" value="' + + FB.esc_attr( fbForm.fields[ id ].options[ i ] ) + + '" class="fb-options" /><div>' + ); + } + } + } + } + // Load editor & hide description + hideDesc(); + } catch ( e ) { + if ( debug ) { + console.log( 'loadFieldEditor(): ' + e ); + } + } + } + function parseShortcode( data ) { + try { + // Clean up fields by resetting them + fbForm.fields = {}; + // Add new fields + if ( ! data ) { + fbForm.fields = defaultFields; + } else { + jQuery.each( data.fields, function( index, value ) { + if ( 1 === parseInt( value.required, 10 ) ) { + value.required = 'true'; + } + fbForm.fields[ index ] = value; + } ); + fbForm.to = data.to; + fbForm.subject = data.subject; + } + } catch ( e ) { + if ( debug ) { + console.log( 'parseShortcode(): ' + e ); + } + } + } + function removeOption( optionId ) { + try { + var thisId = jQuery( '#fb-field-id' ).val(); + var thisVal = jQuery( '#fb-option' + optionId ).val(); + var thisType = jQuery( '#fb-new-type' ).val(); + // Remove from right + jQuery( '#fb-option-box-' + optionId ).remove(); + // Remove from preview + if ( thisType === 'radio' ) { + jQuery( '#fb-radio-' + thisId + '-' + optionId ).remove(); + } else if ( 'checkbox-multiple' === thisType ) { + jQuery( '#fb-checkbox-multiple-' + thisId + '-' + optionId ).remove(); + } else { + jQuery( '#fb-' + thisId + '-' + optionId ).remove(); + } + // Remove from fbForm object + var idx = fbForm.fields[ thisId ].options.indexOf( thisVal ); + if ( idx !== -1 ) { + fbForm.fields[ thisId ].options.splice( idx, 1 ); + } + } catch ( e ) { + if ( debug ) { + console.log( 'removeOption(): ' + e ); + } + } + } + function removeOptions() { + try { + var thisId = jQuery( '#fb-field-id' ).val(); + jQuery( '#fb-options' ).hide(); + if ( optionsCache[ thisId ] === undefined ) { + optionsCache[ thisId ] = {}; + } + optionsCache[ thisId ].options = fbForm.fields[ thisId ].options; // Save options in case they change their mind + fbForm.fields[ thisId ].options = []; // Removes all options + } catch ( e ) { + if ( debug ) { + console.log( 'removeOptions(): ' + e ); + } + } + } + function sendShortcodeToEditor() { + try { + // Serialize fields + jQuery( 'div#sortable div.fb-new-fields' ).each( function( index ) { + var thisId = jQuery( this ).attr( 'fieldid' ); + fbForm.fields[ thisId ].order = index; + } ); + // Export to WYSIWYG editor + jQuery.post( ajaxurl, fbForm, function( response ) { + var isVisual = jQuery( '#edButtonPreview', window.parent.document ).hasClass( 'active' ); + /* WP 3.3+ */ + if ( ! isVisual ) { + isVisual = jQuery( '#wp-content-wrap', window.parent.document ).hasClass( 'tmce-active' ); + } + + var win = window.dialogArguments || opener || parent || top; + var currentCode; + if ( isVisual ) { + currentCode = win.tinyMCE.activeEditor.getContent(); + } else { + currentCode = jQuery( '#editorcontainer textarea', window.parent.document ).val(); + /* WP 3.3+ */ + if ( typeof currentCode !== 'string' ) { + currentCode = jQuery( '.wp-editor-area', window.parent.document ).val(); + } + } + var regexp = new RegExp( + '\\[contact-form\\b.*?\\/?\\](?:[\\s\\S]+?\\[\\/contact-form\\])?' + ); + + // Remove new lines that cause BR tags to show up + response = response.replace( /\n/g, ' ' ); + // Convert characters to comma + response = response.replace( /%26#x002c;/g, ',' ); + + // Add new shortcode + if ( currentCode.match( regexp ) ) { + if ( isVisual ) { + win.tinyMCE.activeEditor.execCommand( + 'mceSetContent', + false, + currentCode.replace( regexp, response ) + ); + } else { + // looks like the visual editor is disabled, + // update the contents of the post directly + jQuery( '#content', window.parent.document ).val( + currentCode.replace( regexp, response ) + ); + } + } else { + try { + win.send_to_editor( response ); + } catch ( e ) { + if ( isVisual ) { + win.tinyMCE.activeEditor.execCommand( 'mceInsertContent', false, response ); + } else { + // looks like the visual editor is disabled, + // update the contents of the post directly + jQuery( '#content', window.parent.document ).val( currentCode + response ); + } + } + } + hidePopup(); + } ); + } catch ( e ) { + if ( debug ) { + console.log( 'sendShortcodeToEditor(): ' + e ); + } + } + } + function showDesc() { + jQuery( '#fb-desc' ).show(); + jQuery( '#fb-add-field' ).hide(); + } + function showAndHideMessage( message ) { + try { + var newMessage = ! message ? GrunionFB_i18n.savedMessage : message; + jQuery( '#fb-success' ).text( newMessage ); + jQuery( '#fb-success' ).slideDown( 'fast' ); + setTimeout( function() { + jQuery( '#fb-success' ).slideUp( 'fast' ); + }, 2500 ); + } catch ( e ) { + if ( debug ) { + console.log( 'showAndHideMessage(): ' + e ); + } + } + } + function switchTabs( whichType ) { + try { + if ( whichType === 'preview' ) { + if ( ! validateEmails( jQuery( '#fb-field-my-email' ).val() ) ) { + return; + } + jQuery( '#tab-preview a' ).addClass( 'current' ); + jQuery( '#tab-settings a' ).removeClass( 'current' ); + jQuery( '#fb-preview-form, #fb-desc' ).show(); + jQuery( '#fb-email-settings, #fb-email-desc' ).hide(); + showAndHideMessage( GrunionFB_i18n.savedMessage ); + } else { + jQuery( '#tab-preview a' ).removeClass( 'current' ); + jQuery( '#tab-settings a' ).addClass( 'current' ); + jQuery( '#fb-preview-form, #fb-desc, #fb-add-field' ).hide(); + jQuery( '#fb-email-settings, #fb-email-desc' ).show(); + jQuery( '#fb-field-my-email' ) + .focus() + .select(); + } + } catch ( e ) { + if ( debug ) { + console.log( 'switchTabs(): ' + e ); + } + } + } + function validateEmails( emails ) { + // Field is allowed to be empty :) + if ( 0 === emails.length ) { + return true; + } + + var $e, + emailList = emails.split( ',' ); + + for ( $e = 0; $e < emailList.length; $e++ ) { + if ( false === validateEmail( emailList[ $e ] ) ) { + alert( emailList[ $e ] + GrunionFB_i18n.invalidEmail ); + return false; + } + } + + return true; + } + /* Uses The Official Standard: RFC 5322 -- http://www.regular-expressions.info/email.html */ + function validateEmail( email ) { + var re = /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/i; + return re.test( email ); + } + function updateLabel() { + try { + var thisId = jQuery( '#fb-field-id' ).val(); + var thisLabel = jQuery( '#fb-new-label' ).val(); + // Update preview + if ( thisLabel.length === 0 ) { + jQuery( '#fb-new-field' + thisId + ' label .label-text' ).text( GrunionFB_i18n.newLabel ); + } else { + jQuery( '#fb-new-field' + thisId + ' label .label-text' ).text( thisLabel ); + } + // Update fbForm object + fbForm.fields[ thisId ].label = thisLabel; + } catch ( e ) { + if ( debug ) { + console.log( 'updateLabel(): ' + e ); + } + } + } + function updateMyEmail() { + try { + var thisEmail = jQuery( '#fb-field-my-email' ).val(); + fbForm.to = thisEmail; + } catch ( e ) { + if ( debug ) { + console.log( 'updateMyEmail(): ' + e ); + } + } + } + function updateOption( that ) { + try { + var thisId = jQuery( '#fb-field-id' ).val(); + var thisOptionid = that.attr( 'optionid' ); + var thisOptionValue = that.val(); + var thisType = jQuery( '#fb-new-type' ).val(); + // Update preview + if ( thisType === 'radio' ) { + jQuery( '#fb-radio-' + thisId + '-' + thisOptionid + ' span' ).text( thisOptionValue ); + } else if ( 'checkbox-multiple' === thisType ) { + jQuery( '#fb-checkbox-multiple-' + thisId + '-' + thisOptionid + ' span' ).text( + thisOptionValue + ); + } else { + jQuery( '#fb-' + thisId + '-' + thisOptionid ).text( thisOptionValue ); + } + // Update fbForm object + fbForm.fields[ thisId ].options[ thisOptionid ] = thisOptionValue; + } catch ( e ) { + if ( debug ) { + console.log( 'updateOption(): ' + e ); + } + } + } + function updateRequired() { + try { + var thisId = jQuery( '#fb-field-id' ).val(); + var thisChecked = jQuery( '#fb-new-required' ).is( ':checked' ); + // Update object and preview + if ( thisChecked ) { + fbForm.fields[ thisId ].required = true; + jQuery( '#fb-new-field' + thisId + ' label' ).append( + '<span class="label-required">' + GrunionFB_i18n.requiredLabel + '</span>' + ); + } else { + fbForm.fields[ thisId ].required = false; + jQuery( '#fb-new-field' + thisId + ' label .label-required' ).remove(); + } + } catch ( e ) { + if ( debug ) { + console.log( 'updateRequired(): ' + e ); + } + } + } + function updateSubject() { + try { + var thisSubject = jQuery( '#fb-field-subject' ).val(); + fbForm.subject = thisSubject; + } catch ( e ) { + if ( debug ) { + console.log( 'updateSubject(): ' + e ); + } + } + } + function updateType( thisType, thisLabelText, thisRequired ) { + try { + var thisId = jQuery( '#fb-field-id' ).val(); + if ( ! thisType ) { + thisType = jQuery( '#fb-new-type' ).val(); + } + if ( ! thisLabelText ) { + thisLabelText = jQuery( '#fb-new-field' + thisId + ' .label-text' ).text(); + } + var isRequired = thisRequired + ? '<span class="label-required">' + GrunionFB_i18n.requiredLabel + '</span>' + : ''; + var thisLabel = + '<label fieldid="' + + thisId + + '" for="fb-field' + + thisId + + '"><span class="label-text">' + + FB.esc_html( thisLabelText ) + + '</span>' + + isRequired + + '</label>'; + var thisRadio = + '<input type="radio" name="radio-' + + thisId + + '" id="fb-field' + + thisId + + ' "disabled="disabled" />'; + var thisRadioLabel = + '<label fieldid="' + + thisId + + '" for="fb-field' + + thisId + + '" class="fb-radio-label"><span class="label-text">' + + FB.esc_html( thisLabelText ) + + '</span>' + + isRequired + + '</label>'; + var thisRadioRemove = '<div class="fb-remove fb-remove-small" id="' + thisId + '"></div>'; + var thisRemove = '<div class="fb-remove" id="' + thisId + '"></div>'; + var thisCheckbox = + '<input type="checkbox" id="fb-field' + thisId + '" "disabled="disabled" />'; + var thisCheckboxMultiple = + '<input type="checkbox" id="fb-field' + thisId + '" "disabled="disabled" />'; + var thisCheckboxMultipleRemove = + '<div class="fb-remove fb-remove-small" id="' + thisId + '"></div>'; + var thisText = '<input type="text" id="fb-field' + thisId + '" "disabled="disabled" />'; + var thisTextarea = '<textarea id="fb-field' + thisId + '" "disabled="disabled"></textarea>'; + var thisClear = '<div class="clear"></div>'; + var thisSelect = + '<select id="fb-field' + + thisId + + '" fieldid="' + + thisId + + '"><option id="fb-' + + thisId + + '-' + + optionsCount + + '" value="' + + thisId + + '-' + + optionsCount + + '">' + + GrunionFB_i18n.firstOptionLabel + + '</option></select>'; + switch ( thisType ) { + case 'checkbox': + removeOptions(); + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRadioRemove + thisCheckbox + thisRadioLabel + thisClear + ); + break; + case 'checkbox-multiple': + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisLabel + + thisCheckboxMultipleRemove + + '<div fieldid="' + + thisId + + '" id="fb-custom-checkbox-multiple' + + thisId + + '"></div>' + ); + if ( + optionsCache[ thisId ] !== undefined && + optionsCache[ thisId ].options.length !== 0 + ) { + fbForm.fields[ thisId ].options = optionsCache[ thisId ].options; + jQuery( '#fb-custom-checkbox-multiple' + thisId ).append( + customOptions( thisId, thisType ) + ); + } else { + jQuery( '#fb-new-options' ).html( + '<label for="fb-option0">' + + GrunionFB_i18n.optionsLabel + + '</label><input type="text" id="fb-option0" optionid="0" value="' + + GrunionFB_i18n.firstOptionLabel + + '" class="fb-options" />' + ); + jQuery( '#fb-custom-checkbox-multiple' + thisId ).append( + '<div id="fb-checkbox-multiple-' + + thisId + + '-0">' + + thisCheckboxMultiple + + '<span>' + + GrunionFB_i18n.firstOptionLabel + + '</span>' + + thisClear + + '</div>' + ); + fbForm.fields[ thisId ].options[ optionsCount ] = GrunionFB_i18n.firstOptionLabel; + } + jQuery( '#fb-options' ).show(); + setTimeout( function() { + jQuery( '#fb-option0' ) + .focus() + .select(); + }, 100 ); + break; + case 'email': + removeOptions(); + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRemove + thisLabel + thisText + ); + break; + case 'name': + removeOptions(); + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRemove + thisLabel + thisText + ); + break; + case 'radio': + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisLabel + + thisRadioRemove + + '<div fieldid="' + + thisId + + '" id="fb-custom-radio' + + thisId + + '"></div>' + ); + if ( + optionsCache[ thisId ] !== undefined && + optionsCache[ thisId ].options.length !== 0 + ) { + fbForm.fields[ thisId ].options = optionsCache[ thisId ].options; + jQuery( '#fb-custom-radio' + thisId ).append( customOptions( thisId, thisType ) ); + } else { + jQuery( '#fb-new-options' ).html( + '<label for="fb-option0">' + + GrunionFB_i18n.optionsLabel + + '</label><input type="text" id="fb-option0" optionid="0" value="' + + GrunionFB_i18n.firstOptionLabel + + '" class="fb-options" />' + ); + jQuery( '#fb-custom-radio' + thisId ).append( + '<div id="fb-radio-' + + thisId + + '-0">' + + thisRadio + + '<span>' + + GrunionFB_i18n.firstOptionLabel + + '</span>' + + thisClear + + '</div>' + ); + fbForm.fields[ thisId ].options[ optionsCount ] = GrunionFB_i18n.firstOptionLabel; + } + jQuery( '#fb-options' ).show(); + setTimeout( function() { + jQuery( '#fb-option0' ) + .focus() + .select(); + }, 100 ); + break; + case 'select': + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRemove + thisLabel + thisSelect + ); + if ( + optionsCache[ thisId ] !== undefined && + optionsCache[ thisId ].options.length !== 0 + ) { + fbForm.fields[ thisId ].options = optionsCache[ thisId ].options; + jQuery( '#fb-field' + thisId ).html( customOptions( thisId, thisType ) ); + } else { + jQuery( '#fb-new-options' ).html( + '<label for="fb-option0">' + + GrunionFB_i18n.optionsLabel + + '</label><input type="text" id="fb-option0" optionid="0" value="' + + GrunionFB_i18n.firstOptionLabel + + '" class="fb-options" />' + ); + fbForm.fields[ thisId ].options[ optionsCount ] = GrunionFB_i18n.firstOptionLabel; + } + jQuery( '#fb-options' ).show(); + setTimeout( function() { + jQuery( '#fb-option0' ) + .focus() + .select(); + }, 100 ); + break; + case 'text': + removeOptions(); + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRemove + thisLabel + thisText + ); + break; + case 'textarea': + removeOptions(); + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRemove + thisLabel + thisTextarea + ); + break; + case 'url': + removeOptions(); + jQuery( '#fb-new-field' + thisId + ' .fb-fields' ).html( + thisRemove + thisLabel + thisText + ); + break; + } + // update object + fbForm.fields[ thisId ].type = thisType; + } catch ( e ) { + if ( debug ) { + console.log( 'updateType(): ' + e ); + } + } + } + return { + resizePop: function() { + try { + //Thickbox won't resize for some reason, we are manually doing it here + var totalWidth = jQuery( 'body', window.parent.document ).width(); + var totalHeight = jQuery( 'body', window.parent.document ).height(); + var isIE6 = typeof document.body.style.maxHeight === 'undefined'; + + jQuery( '#TB_window, #TB_iframeContent', window.parent.document ).css( 'width', '768px' ); + jQuery( '#TB_window', window.parent.document ).css( { + left: ( totalWidth - 768 ) / 2 + 'px', + top: '23px', + position: 'absolute', + marginLeft: '0', + } ); + if ( ! isIE6 ) { + // take away IE6 + jQuery( '#TB_window, #TB_iframeContent', window.parent.document ).css( + 'height', + totalHeight - 73 + 'px' + ); + } + } catch ( e ) { + if ( debug ) { + console.log( 'resizePop(): ' + e ); + } + } + }, + init: function() { + // Scroll to top of page + window.parent.scroll( 0, 0 ); + //Check for existing form data + var contentSource; + if ( + jQuery( '#edButtonPreview', window.parent.document ).hasClass( 'active' ) || + jQuery( '#wp-content-wrap', window.parent.document ).hasClass( 'tmce-active' ) + ) { + var win = window.dialogArguments || opener || parent || top; + contentSource = win.tinyMCE.activeEditor.getContent(); + } else { + contentSource = jQuery( '#content', window.parent.document ).val(); + } + var data = { + action: 'grunion_shortcode_to_json', + _ajax_nonce: ajax_nonce_json, + post_id: postId, + content: contentSource, + }; + + var $doc = jQuery( document ); + + jQuery.post( ajaxurl, data, function( response ) { + // Setup fbForm + parseShortcode( jQuery.parseJSON( response ) ); + // Now build out the preview form + buildPreview(); + } ); + // actions + jQuery( '.fb-add-field' ).click( function() { + addField(); + hideDesc(); + return false; + } ); + jQuery( '#fb-new-label' ).keyup( function() { + updateLabel(); + } ); + jQuery( '#fb-new-type' ).change( function() { + updateType(); + } ); + jQuery( '#fb-new-required' ).click( function() { + updateRequired(); + } ); + $doc.on( 'click', '.fb-remove', function() { + showDesc(); + deleteField( jQuery( this ) ); + grabShortcode(); + } ); + jQuery( '#fb-preview' ).submit( function() { + sendShortcodeToEditor(); + return false; + } ); + jQuery( '#TB_overlay, #TB_closeWindowButton', window.parent.document ).mousedown( function() { + if ( confirm( GrunionFB_i18n.exitConfirmMessage ) ) { + hidePopup(); + } + } ); + $doc.on( 'click', '#fb-another-option', function() { + addOption(); + } ); + $doc.on( 'keyup', '.fb-options', function() { + updateOption( jQuery( this ) ); + } ); + $doc.on( 'click', '.fb-remove-option', function() { + removeOption( jQuery( this ).attr( 'optionid' ) ); + } ); + jQuery( '#tab-preview a' ).click( function() { + switchTabs( 'preview' ); + return false; + } ); + jQuery( '#fb-prev-form' ).click( function() { + switchTabs( 'preview' ); + return false; + } ); + jQuery( '#tab-settings a' ).click( function() { + switchTabs(); + return false; + } ); + jQuery( '#fb-field-my-email' ).blur( function() { + updateMyEmail(); + } ); + jQuery( '#fb-field-subject' ).blur( function() { + updateSubject(); + } ); + $doc.on( 'mouseenter', '.fb-form-case .fb-new-fields', function() { + hideShowEditLink( 'show', jQuery( this ) ); + } ); + $doc.on( 'mouseleave', '.fb-form-case .fb-new-fields', function() { + hideShowEditLink( 'hide' ); + return false; + } ); + $doc.on( 'click', '.fb-edit-field', function() { + editField( jQuery( this ) ); + return false; + } ); + $doc.on( 'click', '.fb-edit-field .fb-reorder', function() { + return false; + } ); + $doc.on( 'click', '#fb-save-field', function() { + showDesc(); + showAndHideMessage(); + return false; + } ); + jQuery( '#fb-feedback' ).click( function() { + var thisHref = jQuery( this ).attr( 'href' ); + window.parent.location = thisHref; + return false; + } ); + jQuery( '#sortable' ).sortable( { + axis: 'y', + handle: '.fb-reorder', + revert: true, + start: function() { + jQuery( '.fb-edit-field' ).hide(); + }, + } ); + jQuery( '#draggable' ).draggable( { + axis: 'y', + handle: '.fb-reorder', + connectToSortable: '#sortable', + helper: 'clone', + revert: 'invalid', + } ); + }, + }; +} )(); diff --git a/plugins/jetpack/modules/contact-form/js/tinymce-plugin-form-button.js b/plugins/jetpack/modules/contact-form/js/tinymce-plugin-form-button.js new file mode 100644 index 00000000..8104ea32 --- /dev/null +++ b/plugins/jetpack/modules/contact-form/js/tinymce-plugin-form-button.js @@ -0,0 +1,37 @@ +/* global grunionEditorView, tinymce */ +( function() { + tinymce.create( 'tinymce.plugins.grunion_form', { + init: function( editor ) { + editor.addButton( 'grunion', { + title: grunionEditorView.labels.tinymce_label, + cmd: 'grunion_add_form', + icon: 'grunion', + } ); + editor.addCommand( 'grunion_add_form', function() { + if ( grunionEditorView.default_form ) { + editor.execCommand( + 'mceInsertContent', + 0, + '[contact-form]' + grunionEditorView.default_form + '[/contact-form]' + ); + } else { + editor.execCommand( 'mceInsertContent', 0, '[contact-form /]' ); + } + } ); + }, + + createControl: function() { + return null; + }, + + getInfo: function() { + return { + longname: 'Grunion Contact Form', + author: 'Automattic', + version: '1', + }; + }, + } ); + + tinymce.PluginManager.add( 'grunion_form', tinymce.plugins.grunion_form ); +} )(); |