Как закончить отрывок предложением, а не словом?

15

Вот выдержка из поста на моей домашней странице. Я хочу закончить выдержку предложением. В данном случае это «возможности». ». Я знаю, как изменить длину, но она не будет делать то, что я хочу.

введите описание изображения здесь

Вот выдержка из другого сайта. Здесь вы видите отрывок, заканчивающийся правильным предложением, в отличие от приведенного выше, которое заканчивается первым словом следующего предложения «Упреждающе».

введите описание изображения здесь

После внедрения решения GM : -

введите описание изображения здесь

Мохит
источник
Пожалуйста, объясните вашу проблему лучше, чем это. Из ответов и комментариев к ним трудно понять, чего вы хотите достичь.
Кайзер
@kaiser Я обновил детали.
Мохит
@ Мохит, ты пробовал мой код?
gmazzap
Да, выдержка заканчивается предложением, но «Читать дальше» больше нет.
Мохит
@GM Хорошо, я должен добавить перевод в ответ.
Мохит

Ответы:

13

Это требует PHP 5.3+ (WP требует PHP 5.2.4+)

add_filter('get_the_excerpt', 'end_with_sentence');

function end_with_sentence($excerpt) {
  $allowed_end = array('.', '!', '?', '...');
  $exc = explode( ' ', $excerpt );
  $found = false;
  $last = '';
  while ( ! $found && ! empty($exc) ) { 
    $last = array_pop($exc);
    $end = strrev( $last );
    $found = in_array( $end{0}, $allowed_end );
  }
  return (! empty($exc)) ? $excerpt : rtrim(implode(' ', $exc) . ' ' .$last);
}

редактировать

После комментария @kaiser я пытаюсь запустить это при сохранении / обновлении, чтобы предотвратить замедление страницы при отображении большого количества сообщений. Это должно быть проверено

add_filter('wp_insert_post_data', 'end_with_sentence_on_save', 20, 2);

function end_with_sentence_on_save($data, $postarr) {
  if ( ! empty( $data['post_content'] ) && $data['post_status'] != 'inherit' && $data['post_status'] != 'trash' ) {
    $text = strip_shortcodes( $data['post_content'] );
    $text = apply_filters('the_content', $text );
    $text = str_replace(']]>', ']]>', $text );
    $excerpt_length = apply_filters('excerpt_length', 55);
    $data['post_excerpt'] = wp_trim_words($text, $excerpt_length, '');
  } else {
    return $data;
  }
  $allowed_end = array('.', '!', '?', '...');
  $exc = explode(' ', $data['post_excerpt']);
  $found = false;
  $last = '';
  while ( ! $found && ! empty($exc) ) { 
    $last = array_pop($exc);
    $end = strrev( $last );
    $found = in_array( $end{0}, $allowed_end );
  }
  if (! empty($exc)) $data['post_excerpt'] = rtrim(implode(' ', $exc) . ' ' .$last);
  return $data; 
}
Gmazzap
источник
1
уверен , что: $end{0}? Кстати, вы можете просто вернуться или breakкогда что-то будет найдено. Дополнительное примечание: это может быть лучше при сохранении поста, поэтому он не регенерируется во время выполнения, что может быть довольно медленным, если вы отображаете много постов.
Кайзер
@kaiser cite: для доступа к строкам также можно использовать фигурные скобки, как в $ str {42}, для той же цели. php.net docs Чтобы найти что-то, мне нужно было найти последний символ последней строки. Используя $foundя разрываю цикл и проверяю последнюю строку, так что нет необходимости break.
gmazzap
Полностью пропустил, что это whileпроверка false. +1
кайзер
@kaiser +1 для второй части вашего комментария, теперь я редактирую код для запуска при сохранении
gmazzap
1
Конечно, @jessica был плохим редактором (в оригинальной версии все было в порядке). Спасибо, исправлено.
gmazzap
2

Вот моя версия, сохраняя все html-теги, а также обрезая содержимое после последнего слова в предложении

if ( ! function_exists( 'pietergoosen_custom_wp_trim_excerpt' ) ) : 

    function pietergoosen_custom_wp_trim_excerpt($pietergoosen_excerpt) {
    global $post;
    $raw_excerpt = $pietergoosen_excerpt;
        if ( '' == $pietergoosen_excerpt ) {

            $pietergoosen_excerpt = get_the_content('');
            $pietergoosen_excerpt = strip_shortcodes( $pietergoosen_excerpt );
            $pietergoosen_excerpt = apply_filters('the_content', $pietergoosen_excerpt);
            $pietergoosen_excerpt = str_replace(']]>', ']]>', $pietergoosen_excerpt);

            //Set the excerpt word count and only break after sentence is complete.
                $excerpt_word_count = 75;
                $excerpt_length = apply_filters('excerpt_length', $excerpt_word_count); 
                $tokens = array();
                $excerptOutput = '';
                $count = 0;

                // Divide the string into tokens; HTML tags, or words, followed by any whitespace
                preg_match_all('/(<[^>]+>|[^<>\s]+)\s*/u', $pietergoosen_excerpt, $tokens);

                foreach ($tokens[0] as $token) { 

                    if ($count >= $excerpt_word_count && preg_match('/[\?\.\!]\s*$/uS', $token)) { 
                    // Limit reached, continue until  ? . or ! occur at the end
                        $excerptOutput .= trim($token);
                        break;
                    }

                    // Add words to complete sentence
                    $count++;

                    // Append what's left of the token
                    $excerptOutput .= $token;
                }

            $pietergoosen_excerpt = trim(force_balance_tags($excerptOutput));

                $excerpt_end = ' <a href="'. esc_url( get_permalink() ) . '">' . '&nbsp;&raquo;&nbsp;' . sprintf(__( 'Read more about: %s &nbsp;&raquo;', 'pietergoosen' ), get_the_title()) . '</a>'; 
                $excerpt_more = apply_filters('excerpt_more', ' ' . $excerpt_end); 

                //$pos = strrpos($pietergoosen_excerpt, '</');
                //if ($pos !== false)
                // Inside last HTML tag
                //$pietergoosen_excerpt = substr_replace($pietergoosen_excerpt, $excerpt_end, $pos, 0);
                //else
                // After the content
                $pietergoosen_excerpt .= $excerpt_end;

            return $pietergoosen_excerpt;   

        }
        return apply_filters('pietergoosen_custom_wp_trim_excerpt', $pietergoosen_excerpt, $raw_excerpt);
    }

endif; 

remove_filter('get_the_excerpt', 'wp_trim_excerpt');
add_filter('get_the_excerpt', 'pietergoosen_custom_wp_trim_excerpt'); 
Питер Гусен
источник
0

На всякий случай, если кто-то использует ответ, совет: это разрушит Расширенные пользовательские поля. ACF также вызывается wp_insert_post_dataпри создании или обновлении поля, но $dataпеременная не совпадает, и в результате имя поля ACF становится нечитаемым бредом, что приводит к невозможности использования поля.

Эту Пуртонен
источник