Кнопка "Читать дальше" не сработает при нажатии на нее один раз, а много раз

Это небольшая проблема, но я застрял. У меня есть кнопка Подробнее для каждого сообщения на странице, которая не сработает, если щелкнуть один раз. Он работает, щелкая по нему столько раз, сколько сообщений на странице.

Вот мой код:

$(function(){
    var maxLength = 500;
    $(".content-wrapper").each(function(){
        var myStr = $(this).html();
        if ($.trim(myStr).length > maxLength){
            var newStr = myStr.substring(0, maxLength);
            var removedStr = myStr.substring(maxLength, $.trim(myStr).length);
            $(this).empty().html(newStr);
            $(this).append(' <a href = "javascript:void(0);" class = "read-more">read more...</a>');
            $(this).append('<span class = "more-text">' + removedStr + '</span>');
        }
    });
   
});

 $(".read-more").click(function(event){
       $(this).hide();
        $(this).siblings(".more-text").show();
        event.preventDefault();
    });
            .content-wrapper .more-text{
                display: none;
            }
       
<div class = "content-wrapper">
            <span style = "color:green"><h4>{{TITLE}}</h4> </span>
            {{POST}}
        </div>
       
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
В настоящее время производительность загрузки веб-сайта имеет решающее значение не только для удобства пользователей, но и для ранжирования в...
Безумие обратных вызовов в javascript [JS]
Безумие обратных вызовов в javascript [JS]
Здравствуйте! Юный падаван 🚀. Присоединяйся ко мне, чтобы разобраться в одной из самых запутанных концепций, когда вы начинаете изучать мир...
Система управления парковками с использованием HTML, CSS и JavaScript
Система управления парковками с использованием HTML, CSS и JavaScript
Веб-сайт по управлению парковками был создан с использованием HTML, CSS и JavaScript. Это простой сайт, ничего вычурного. Основная цель -...
JavaScript Вопросы с множественным выбором и ответы
JavaScript Вопросы с множественным выбором и ответы
Если вы ищете платформу, которая предоставляет вам бесплатный тест JavaScript MCQ (Multiple Choice Questions With Answers) для оценки ваших знаний,...
0
0
44
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Ответ принят как подходящий

jQuery знает только об элементах на странице во время ее выполнения, поэтому новые элементы, добавленные в DOM, не распознаются jQuery. Чтобы решить эту проблему, используйте делегирование мероприятия, восходящую цепочку событий от вновь добавленных элементов до точки в DOM, которая была там, когда jQuery запускался при загрузке страницы. Многие люди используют document как место для перехвата всплывающего события, но нет необходимости продвигаться вверх по дереву DOM. В идеале вы должны передать полномочия ближайшему родителю, существующему на момент загрузки страницы.

$(document).on('click','read-more', function(event){
    $(this).hide();
    $(this).siblings(".more-text").show();
    event.preventDefault();
});

Вы можете делегировать $('content-wrapper')

Обновлено: вы неправильно скрывали дополнительный контент. Вот весь код:

$(function() {

    var maxLength = 500;
    $(".content-wrapper").each(function() {
      var myStr = $(this).html();

      if ($.trim(myStr).length > maxLength) {
        var newStr = myStr.substring(0, maxLength);

        var removedStr = myStr.substring(maxLength, $.trim(myStr).length);
        $(this).empty().html(newStr);
        $(this).append(' <a href = "javascript:void(0);" class = "read-more">read more...</a>');
        $(this).append('<span class = "more-text">' + removedStr + '</span>');
        $('.more-text').hide(); // <-- line added
      }
    });

    $('.content-wrapper').on('click', '.read-more', function(event) {
       $(this).hide();
       $(this).siblings(".more-text").show();
       event.preventDefault();
    });
})

ПРИМЕР

Этот код работает так же, как и то, что я опубликовал.

Wannabe JavaGeek 10.09.2018 18:07

Ошибок нет. Я не знаю, почему это не работает.

Wannabe JavaGeek 10.09.2018 18:09

Это все кода jQuery, который у вас есть на странице? Включили ли вы в проект библиотеку jQuery?

Jay Blanchard 10.09.2018 18:10

да, это все, что у меня есть, и библиотека jquery включена

Wannabe JavaGeek 10.09.2018 18:15

У вас есть дополнительный </SPAN> в каждом div, от которого вам нужно избавиться.

Jay Blanchard 10.09.2018 18:16

Позвольте нам продолжить обсуждение в чате.

Wannabe JavaGeek 10.09.2018 18:21

Я отредактировал ответ и протестировал код с вашей разметкой

Jay Blanchard 10.09.2018 18:27

Рад помочь @WannabeJavaGeek

Jay Blanchard 10.09.2018 19:15

Вы можете сделать, как сказал Джей Бланчард, или вы можете просто прикрепить прослушиватели событий щелчка для элементов класса read-more после добавления таким образом, в исходном коде вы прикрепляли события щелчка к элементам с именем класса read-more сразу после загрузки файла сценария, они делают не существует в то время, если вы переместите прикрепление событий щелчка к элементам с className read-more внутри документа. уже после их добавления они будут работать:

$(function(){
    var maxLength = 500;
    $(".content-wrapper").each(function(){
        var myStr = $(this).html();
        if ($.trim(myStr).length > maxLength){
            var newStr = myStr.substring(0, maxLength);
            var removedStr = myStr.substring(maxLength, $.trim(myStr).length);
            $(this).empty().html(newStr);
            $(this).append(' <a href = "javascript:void(0);" class = "read-more">read more...</a>');
            $(this).append('<span class = "more-text">' + removedStr + '</span>');
        }
    });

    //Now that the read-more className elements exist in the DOM from .append
    //Attach the click event handlers
    $(".read-more").click(function(event){
       $(this).hide();
        $(this).siblings(".more-text").show();
        event.preventDefault();
    });



});

@JayBlanchard В коде OP у них есть код, работающий за пределами document.ready, поэтому он запускается до изменения DOM и включает добавленные элементы с className `read-more ', в этом ответе прикрепление щелчка события выполняются после того, как добавление выполнено в DOM внутри документа.

Ryan Wilson 10.09.2018 17:57

Вы должны объяснить это в своем ответе, поскольку это не очевидно.

Jay Blanchard 10.09.2018 17:58

@JayBlanchard Обновил мой ответ, чтобы он стал более понятным.

Ryan Wilson 10.09.2018 18:00

@WannabeJavaGeek Это не то же самое, и я бы посоветовал вам прочитать комментарий перед ответом, чтобы понять, чем он отличается.

Ryan Wilson 10.09.2018 18:13

Другие вопросы по теме