Tooltipster: [data-tooltip-content] как указатель на Dynami c HTML - PullRequest
0 голосов
/ 12 марта 2020

ПРОБЛЕМА : Использование Tooltipster. JS с FullCalendar и / или с динамической визуализацией content

Я долго царапал голову и пережевывал это, и я не близко к тому месту, где я был, когда начинал. Я прочитал и перечитал документы Tooltipster, но я просто не могу использовать 'data-tooltip-content' в качестве указателя на #idTooltipsterElement. Он просто отображает значение 'data-tooltip-content', хотя в документах четко указано:

Состояния документации Tooltipster:

5. Используйте HTML внутри ваших подсказок

Tooltipster позволяет вам использовать любую HTML разметку внутри ваших подсказок. Это означает, что вы можете вставлять такие вещи, как изображения и теги форматирования текста.

Вместо атрибута title используйте атрибут data-tooltip-content, чтобы обеспечить селектор, соответствующий элементу HTML вашей страницы, который должен использоваться как контент. Это ваш HTML:

<span class="tooltip" data-tooltip-content="#tooltip_content">This span has a tooltip with HTML when you hover over it!</span>

<div class="tooltip_templates">
    <span id="tooltip_content">
        <img src="myimage.png" /> <strong>This is the content of my tooltip!</strong>
    </span>
</div>

В вашем файле CSS добавьте .tooltip_templates { display: none; }, чтобы содержимое не отображалось вне подсказки.

Когда мы перешли с Типпи. js на Tooltipster. js У меня были другие проблемы, которых, к счастью, нет в Tootipster, но я нашел Первый был гораздо проще в настройке, хотя он не был близок ни к функциональности, ни к документации, как к последнему.

Однако: ранее я просто использовал 'data-tooltip-content', чтобы поместить в него все свои элементы HTML, и как ни странно, я заметил, что это также работает с Tooltipster , хотя их документация кажется предпочитаю вышеуказанный подход. Теперь обычно я просто сохраняю старый подход, с которым я уже знаком, но, к сожалению, он приносит с собой целый ряд потенциальных проблем , связанных с изменениями рендеринга после события (которые мы, вероятно, реализуем) в FullCalendar которые мы широко используем на наших страницах, а также форматирование / трудности с оформлением , которые я уже испытывал.

Я разумно Конечно, предпочтительное решение , описанное в документированном Tooltipster, было бы идеальным для нашей реализации, но я просто не могу понять, как заставить его работать в нашей среде fullCalendar , которая разбита в следующем потоке программы:

  1. eventRender : извлекать события из источника событий и отображать каждую запись для выбранного месяца / периода. Это где первоначальный Toolstipster создается и ему присваивается уникальный идентификатор

  2. eventAfterAllRender : это в основном выполняется после eventRender, и именно здесь мы выполняем пост-рендеринг Например, проверьте наличие недействительных ссылок на изображения и т. д. c и замените, удалите все значки / ссылки на изображения, а также измените или отключите / удалите всплывающую подсказку, если она больше не нужна. При необходимости мы заменяем здесь содержимое всплывающей подсказки, в противном случае мы устанавливаем для содержимого всплывающей подсказки все данные, которые были переданы из процесса eventRender.

  3. eventMouseover : этот раздел содержит событие метод hover, который отвечает за отображение всплывающей подсказки Tooltipster.

Я создал образец скрипта, используя сокращенную версию нашего кода и пару тестовых событий вместо использования внешнего источника данных, но результат тот же, когда я использую этот код в нашей собственной среде. Я включил следующее, чтобы отразить нашу среду: Bootstrap 4.4.0 - FullCalendar 3.10.1 - Tooltipster 4.2.7

Если навести курсор на одно из событий, всплывающая подсказка просто отображает значение указателя раздела, например, #tt_event1 (идентификатор элемента, на который он указывает) вместо того, что установлено в разделе содержимого

, например <div class="tooltip_templates"><span id="tt_event1" class="tooltip_content"><img src="https://hackernoon.com/hn-images/1*cmqZiJz1TuUedRoeI3g_Iw.jpeg" width="450" height="auto"><p style="text-align:left;"><span class="flag-icon flag-icon-uk"></span><strong class="title">Tips for Writing Cleaner Code</strong><br>optional desctiptive text can go here</p></span></div>

  $(document).ready(function() {

    $('#calendar').fullCalendar({
      defaultView: 'month',
      header: '',
      defaultDate: '2020-03-01',
      events: [{
        id: 'event1',
        className: 'UK',
        title: 'Tips for Writing Cleaner Code',
        description: 'I decided to write an article that will be useful for beginners to understand their mistakes and to put together some practices. /n source: hackernoon.com/tips-for-writing-cleaner-better-code-e36ffeb55526',
        start: '2020-03-02', end: '2020-03-02'
      }, {
        id: 'event2',
        className: 'NL',
        title: 'Modern Style of Javascript with Arrow Functions',
        description: 'The complete explanation of Arrow functions in Javascript, and how it helps developers to write flexible and consistent code. /n source: hackernoon.com/modern-style-of-javascript-with-arrow-functions-lg1x3474',
        start: '2020-03-04', end: '2020-03-11'
      }
              ],
      eventRender: function(event, element, view) {

        window.dataE = window.dataE || [];
        element.attr( 'id', event.id );

        var /* Vars */
        desc = (event.description), url='',
            urlEvent1 = "https://hackernoon.com/hn-images/1*cmqZiJz1TuUedRoeI3g_Iw.jpeg",
            urlEvent2 = "https://images.unsplash.com/photo-1527427337751-fdca2f128ce5?ixlib=rb-1.2.1&q=80&fm=jpg&crop=entropy&cs=tinysrgb&w=1080&fit=max&ixid=eyJhcHBfaWQiOjEwMDk2Mn0",
            tipRef="tip_content_"+event.id, idTip="#"+tipRef
        /* End Vars */;

        // passing 2 extra image params manually - usually from seperate data routine
        if (event.title == 'Tips for Writing Cleaner Code') { url = urlEvent1; } else { url = urlEvent2; }

        var tt_ref = '#tt_'+event.id;
        var className = ''+event.className;
        var cc = className.toLowerCase();
        var tt = '<div class="tooltip_templates"><span id="tt_' + event.id + '" class="tooltip_content"><img src="' + url + '" width="450" height="auto"><p style="text-align:left;"><span class="flag-icon flag-icon-' + cc + '"></span><strong class="title">' + event.title + '</strong><br>optional desctiptive text can go here</p></span></div>';

        element.attr( 'data-tooltip-content', tt_ref ); // the tooltip pointer
        element.attr( 'data-tt-tooltipser' ); element.attr( 'data-tt-tooltipser', tt );
        element.attr( 'tt_title' ); element.attr( 'tt_title', event.title );
        element.addClass('tt_tooltip tt_group');

        var eID = '#'+event.id;

        // Tooltipster
        if (!element.hasClass('tt_added')) {
          $(eID).tooltipster();
          $(eID).tooltipster({
            //content: tipContent,
            //contentCloning: true,
            trigger: 'hover',
            //multiple: true,
            animation: 'fade',
            arrow: true,
            delay: 300,
            maxWidth: 600,
            contentAsHTML: true,
            debug: true
          });
          element.addClass('tt_added');
        }

        // create array of event.IDs for use in [eventAfterAllRender]
        if (Array.isArray(dataE)){
          var json = JSON.stringify(event.id),
              item = dataE.find(el => JSON.stringify(el) === json);
          if (typeof item !== 'undefined'){
            return false;
          } else {
            dataE.push(event.id);
          }
        }

      },

      eventAfterAllRender: function(event, element){

        // get events from dataE array created during [eventRender]
        var count = 0;
        for (var i=0; i<dataE.length; i++) {

          var id = dataE[i], eID = '#'+id ;

          // now obtain the tooltip & tooltipster variables for each event
          var tipTooltipRef = $(eID).attr('data-tooltip-content');
          var tipTooltipsterContent = $(eID).attr('data-tt-tooltipser');

          console.log("tipTooltipRef:", tipTooltipRef);
          console.log("tipTooltipsterContent:", tipTooltipsterContent);
          console.log("$(eID)", $(eID));

          // append the tooltipster content aquired via tooltipster var
          $(eID).append(tipTooltipsterContent);
          $(eID).tooltipster();
          // set the content pointer
          $(eID).tooltipster('content', tipTooltipRef );


          //TOOLTIPSTER: update any necessary Tooltip content
          $('.fc-event').mouseenter(function() {

            if (tipTooltipRef == '' || tipTooltipRef == 'undefined'){
              $(eID).tooltipster('content', 'Invalid image link ?');
              // OR simply: $(this).tooltipster('disable');
            } else {
              var t = $(eID).attr('tt_title'), tt_Title = '<div class="ttTitle">'+t+'</div>' ;
              var tt_element = $(eID).find('.tooltipster-content');
              tt_element.append(tt_Title);
              $(eID).tooltipster('option','contentAsHTML','true');
              $(eID).tooltipster('content', tipTooltipRef);

              if (!$(eID).hasClass('tt_added')) {
                $(eID).tooltipster();
                $(eID).tooltipster({
                  content: tipTooltipRef,
                  //contentCloning: true,
                  trigger: 'hover',
                  //multiple: true,
                  animation: 'fade',
                  arrow: true,
                  delay: 300,
                  maxWidth: 600,
                  contentAsHTML: true,
                  //debug: true
                });
              // bind on start events (triggered on mouseenter)
              $(eID).on('start', function(event) {
               if ($(event.instance.elementOrigin()).hasClass('tt_group')){
                    var instances = $.tooltipster.instances('.tt_group'),
                        open = false,
                        duration;
                    $.each(instances, function (i, instance) {
                      if (instance !== event.instance) {
                        // if another instance is already open
                        if (instance.status().open){
                          open = true;
                          // get the current animationDuration
                          duration = instance.option('animationDuration');
                          // close the tooltip without animation
                          instance.option('animationDuration', 0);
                          instance.close();
                       // restore the animationDuration to its normal value
                          instance.option('animationDuration', duration);
                        }
                      }
                    });
                  // if another instance was open
                  if (open) {
                     duration = event.instance.option('animationDuration');
                      // open the tooltip without animation
                      event.instance.option('animationDuration', 0);
                      event.instance.open();
                      // restore the animationDuration to its normal value
                      event.instance.option('animationDuration', duration);
                      // now that we have opened the tooltip,
                      //the hover trigger must be stopped
                      event.stop();
                    }
                  }
                });
                $(eID).addClass('tt_added');
              }
            }
          });

        }

      },

      eventMouseover: function(view, event, element){

        //TOOLTIPSTER: update any necessary Tooltip content
        var tipContent = $(this).attr('data-ttipster');
        var id = event.id //$(this).attr('id');
        var eID = '#'+id;
        var tipID = '#tt_'+id;

        if (tipContent == '' || tipContent == 'undefined'){
          $(eID).tooltipster('content', 'Invalid image ? detected: unable to display at present ?');
          $(eID).tooltipster('disable');
          // or $(this).tooltipster('destroy');
        } else {
  // TOOLTIPSTER: Not really req now as tipContent is set @ evenRender
          //$(eID).tooltipster('option','contentAsHTML','true');
          $(eID).tooltipster('option','multiple','true');
          $(eID).tooltipster({
            functionInit: function(instance, helper){
              var content = $(helper.origin).find(tipID).detach();
              instance.content(content);
            }
          });
          if (!$(this).hasClass('tt_added')) {
            $(eID).tooltipster();
            $(eID).tooltipster({
              content: tipContent,
              //contentCloning: true,
              trigger: 'hover',
              //multiple: true,
              animation: 'fade',
              arrow: true,
              delay: 300, //[300, 100]
              maxWidth: 600,
              contentAsHTML: true,
              debug: true
            });
            // bind on start events (triggered on mouseenter)
            $(this).on('start', function(event) {
              if ($(event.instance.elementOrigin()).hasClass('tt_group')) {
                var instances = $.tooltipster.instances('.tt_group'),
                    open = false,
                    duration;
                $.each(instances, function (i, instance) {
                  if (instance !== event.instance) {
                    // if another instance is already open
                    if (instance.status().open){
                      open = true;
                      // get the current animationDuration
                      duration = instance.option('animationDuration');
                      // close the tooltip without animation
                      instance.option('animationDuration', 0);
                      instance.close();
                      // restore the animationDuration to its normal value
                      instance.option('animationDuration', duration);
                    }
                  }
                });
                // if another instance was open
                if (open) {
                  duration = event.instance.option('animationDuration');
                  // open the tooltip without animation
                  event.instance.option('animationDuration', 0);
                  event.instance.open();
                  // restore the animationDuration to its normal value
                  event.instance.option('animationDuration', duration);
                  // now that we have opened the tooltip,
                  //the hover trigger must be stopped
                  event.stop();
                }
              }
            });
            $(this).addClass('tt_added');
          }
        }

      },

      eventClick:  function(event, element, view) {
        var e = (event.description);
        if (e != null){
          var chr = e.length;
          // event click coded goes here
          alert(e);
        }
      }

    });
  });
/* tooltipster.js */
.tooltip_templates {
  display: none;
}
.tooltipster-content{
  /*display: flex;
  flex-direction: column;*/
}
.ttTitle {
}

/*! suit-flex-embed v1.4.0 | MIT License | github.com/suitcss */
.FlexEmbed {
  display: block;
  overflow: hidden;
  position: relative;
}
.FlexEmbed:before {
  content: "";
  display: block;
  width: 100%;
}

.FlexEmbed--16by9:before {
  padding-bottom: 56.25%;
}
.FlexEmbed--4by3:before {
  padding-bottom: 75%;
}
.FlexEmbed--1by1:before {
  padding-bottom: 100%;
}
.CoverImage {
  background-position: 50%;
  background-repeat: no-repeat;
  background-size: cover;
  margin: 0 auto 1em;
  max-height: 600px;
  max-width: 600px;
}
.CoverImageX2 {
  background-color: #808080;
  background-position: 50%;
  background-repeat: no-repeat;
  background-size: 100% 100%;  /*cover; contain;*/
  margin: 0 auto 1em;
  max-height: 2400px;
  max-width: 1200px;
}
<!-- Bootstrap -->
<script src="https://code.jquery.com/jquery-3.3.1.min.js" integrity="sha256-FgpCb/KJQlLNfOu91ta32o/NMZxltwRo8QtmkMRdAu8=" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/popper.js@1.16.0/dist/umd/popper.min.js" integrity="sha384-Q6E9RHvbIyZFJoft+2mJbHaEWldlvI9IOYy5n3zV9zzTtmI3UksdQRVvoxMfooAo" crossorigin="anonymous"></script>
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.4.0/css/bootstrap.min.css" integrity="sha384-SI27wrMjH3ZZ89r4o+fGIJtnzkAnFs3E4qz9DIYioCQ5l9Rd/7UAa8DHcaL8jkWt" crossorigin="anonymous" />
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.4.0/js/bootstrap.min.js" integrity="sha384-3qaqj0lc6sV/qpzrc1N5DC6i1VRn/HyX4qdPaiEFbn54VjQBEU341pvjz7Dv3n6P" crossorigin="anonymous"></script>

<!-- FulCalendar  -->
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fullcalendar/3.10.1/fullcalendar.min.css" integrity="sha256-tXJP+v7nTXzBaEuLtVup1zxWFRV2jyVemY+Ir6/CTQU=" crossorigin="anonymous" />
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.24.0/moment.min.js" integrity="sha256-4iQZ6BVL4qNKlQ27TExEhBN1HFPvAvAMbFavKKosSWQ=" crossorigin="anonymous"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/fullcalendar/3.10.1/fullcalendar.min.js" integrity="sha256-O04jvi1wzlLxXK6xi8spqNTjX8XuHsEOfaBRbbfUbJI=" crossorigin="anonymous"></script>

<!-- Tooltipster -->
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/css/tooltipster.main.min.css" integrity="sha256-xlmCQ8IjIIx7gqrIAb5x5kEU30jJJm0/DEmrjgLow/E=" crossorigin="anonymous" />
    <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/css/tooltipster.bundle.min.css" integrity="sha256-Qc4lCfqZWYaHF5hgEOFrYzSIX9Rrxk0NPHRac+08QeQ=" crossorigin="anonymous" />
    <script src="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/js/tooltipster.main.min.js" integrity="sha256-9gPC19rdxygnD5cXHFodzczLKeucNZ/dgzLhkKvNtQM=" crossorigin="anonymous"></script>
    <script src="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/js/tooltipster.bundle.min.js" integrity="sha256-NOU7KrY2aTI4PxDegqYUIknk9qfxVCS0E4JfE9aMwaA=" crossorigin="anonymous"></script>
    <script src="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/js/plugins/tooltipster/SVG/tooltipster-SVG.min.js" integrity="sha256-b9JNfGq08bjI5FVdN3ZhjWBSRsOyF6ucACQwlvgVEU4=" crossorigin="anonymous"></script>

<div id='calendar'></div>

В качестве альтернативы, если вы предпочитаете скрипку, которую я сделал ранее: https://jsfiddle.net/magicmb/3manqpho/


ДОПОЛНИТЕЛЬНО: Дополнительный бонусный вопрос

Еще одна потенциальная проблема с нашей реализацией Tooltipster. JS, и, возможно, для этого мне, возможно, придется создать отдельный вариант SO, но я в любом случае, я упомяну об этом здесь, так как я изо всех сил пытаюсь создать для него рабочую скрипку. То, что я пытался сделать со своим вторым событием, это продемонстрировать еще одно немного странное поведение Tooltipster:

Кажется, что Tooltipster имеет сложность с событиями, которые охватывают несколько дней и отличаются строки в FullCalendar. Кажется, что они работают только при наведении на первую часть события, то есть на бит в row1 , а не на наведение над второй частью в row2 .

Теперь, к сожалению, я не смог показать это на моей тестовой скрипке. Причина в том, что вполне может потребоваться создание отдельных событий для каждого дня (элементы в FullCalendar), хотя мое одиночное событие по какой-то причине кажется go до конца первой строки, но не до следующей. Следовательно, я не смог показать причудливое поведение, создав событие, которое охватывает две недели с одной датой начала и одной датой окончания, как я делал в моем тестовом примере.

Наша собственная система использует внешние источники данных , и это никогда не было очевидным. Также он работает только с нашей предыдущей реализацией Tooltip. js, и первоначально я также думал, что она работает с Tooltipster, но сейчас я не уверен. Возможно, это просто проблема с более поздними версиями Tooltipster [ EDIT : на самом деле из-за ранее использовавшегося data-tooltip-content для хранения всей всплывающей подсказки] или, возможно, они работают немного по-другому путь от предыдущих выпусков. В любом случае, если кто-то из вас знает что-то об этом, не стесняйтесь упоминать об этом в дополнение к главной проблеме, приведенной выше, о том, как заставить всплывающую подсказку HTML работать с использованием указателей внутри data-tooltip-content.

1 Ответ

1 голос
/ 19 марта 2020

Через некоторое время обдумывая и пробуя различные вещи, а также ссылаясь на документы Tootltipster & FullCalendar и многочисленные проблемы с Github и др. c Я выяснил свои проблемы.

Ключевой шаг:

Все всплывающие подсказки должны быть созданы как часть процесса FullCalendar [eventRender] , и мы ничего не запускаем через [eventMouseover] . Также вместо привязки всплывающей подсказки Tooltipster к event.id вам необходимо связать ее с классом подсказки или, в моем случае, .tt_tooltip классом из-за «всплывающей подсказки» конфликт имен с Bootstrap.

Еще одна важная вещь, на которую следует обратить внимание, это то, что только кажется, что мы инициализируем каждого из всплывающих подсказок всплывающие подсказки при [eventRender] , а также при настройке параметров, в документации FullCalendar указано:

eventRender

Triggered while an event is being rendered. A hook for modifying its DOM.

    function( event, element, view ) { }

event is the Event Object that is attempting to be rendered.

element is a newly created jQuery element that will be used for rendering.

Дополнительные важные шаг:

Нам нужно выполнить generi c Подсказка всплывающей подсказки Инициировать в конце do c .ready для всех ранее созданных всплывающих подсказок с использованием tt_tooltip класса: $('.tt_tooltip').tooltipster();

РЕДАКТИРОВАТЬ: Согласно моему дополнительному комментарию ниже это просто Например, в моем тестовом / живом окружении мне пришлось звонить $('.tt_tooltip').tooltipster(); в конце [eventAfterAllRender:] вместо do * 107 7 * .ready !

Вот исправленный пример кода:

  $(document).ready(function() {

    $('#calendar').fullCalendar({
      defaultView: 'month',
      header: '',
      defaultDate: '2020-03-01',
      events: [{
        id: 'event1',
        className: 'GB',
        title: 'Tips for Writing Cleaner Code',
        description: 'I decided to write an article that will be useful for beginners to understand their mistakes and to put together some practices. /n source: hackernoon.com/tips-for-writing-cleaner-better-code-e36ffeb55526',
        start: '2020-03-02', end: '2020-03-02'
      }, {
        id: 'event2',
        className: 'NL',
        title: 'Modern Style of Javascript with Arrow Functions',
        description: 'The complete explanation of Arrow functions in Javascript, and how it helps developers to write flexible and consistent code. /n source: hackernoon.com/modern-style-of-javascript-with-arrow-functions-lg1x3474',
        start: '2020-03-04', end: '2020-03-11'
      }
              ],
      eventRender: function(event, element, view) {

        window.dataE = window.dataE || [];
        element.attr( 'id', event.id );
        var eExist = false;

        // create array of event.IDs for use in [eventAfterAllRender]
        if (Array.isArray(dataE)){
          var json = JSON.stringify(event.id),
              item = dataE.find(el => JSON.stringify(el) === json);
          if (typeof item !== 'undefined'){
            return false;
          } else {
            dataE.push(event.id);
            eExist = true;
          }
        }


        var /* Vars */
        desc = (event.description), url='',
            urlEvent1 = "https://hackernoon.com/hn-images/1*cmqZiJz1TuUedRoeI3g_Iw.jpeg",
            urlEvent2 = "https://images.unsplash.com/photo-1527427337751-fdca2f128ce5?ixlib=rb-1.2.1&q=80&fm=jpg&crop=entropy&cs=tinysrgb&w=1080&fit=max&ixid=eyJhcHBfaWQiOjEwMDk2Mn0",
            tipRef="tip_content_"+event.id, idTip="#"+tipRef
        /* End Vars */;

        // passing 2 extra image params manually - usually from seperate data routine
        if (event.title == 'Tips for Writing Cleaner Code') { url = urlEvent1; } else { url = urlEvent2; }

        var tt_ref = '#tt_'+event.id;
        //var tt_ref = 'tt_'+event.id;
        var className = ''+event.className;
        var cc = className.toLowerCase();
        var tt = '<div class="tooltip_templates"><span id="tt_' + event.id + '" class="tooltip_content"><img src="' + url + '" width="350" height="auto"><p style="text-align:left;"><span class="flag-icon flag-icon-' + cc + '"></span> <span class="tt_title"><strong>' + event.title + '</strong></span><br><span class="tt_desc">optional desctiptive text can go here</span></p></span></div>';

        element.attr( 'data-tooltip-content', tt_ref ); // the tooltip ref selector (pointer)
        element.attr( 'data-tt-tooltipster' ); element.attr( 'data-tt-tooltipster', tt ); // tootipster content stored for [AfterEventRender]
        element.attr( 'tt_title' ); element.attr( 'tt_title', event.title ); 
        element.addClass('tt_tooltip'); // add the tooltipster class (using 'tt_tooltip' due to Bootstrap 'tooltip' conflict )
        element.append(tt); // append tooltipster content to each fullCalendar element

        // Tooltipster - as per Tooltipster Doc use 'tooltip' class (in our case 'tt_tooltip' due to Bootstrap conflict)
        $('.tt_tooltip').tooltipster();
        $('.tt_tooltip').tooltipster({
          content: tt_ref,
          contentCloning: true,
          trigger: 'hover',
          multiple: true,
          animation: 'fade',
          arrow: true,
          delay: 300,
          maxWidth: 600,
          contentAsHTML: true,
          debug: true
        });


      },

      eventAfterAllRender: function(event, element){

        // get events from dataE array created during [eventRender]
        var count = 0;
        for (var i=0; i<dataE.length; i++) {

          var id = dataE[i], eID = '#'+id ;
          // obtain the tooltip is selector & tooltipster data each event
          var tipTooltipRef = $('.fc-body').find(eID).closest('.tt_tooltip').attr('data-tooltip-content');
          var tipTooltipsterContent = $('.fc-body').find(eID).closest('.tt_tooltip').attr('data-tt-tooltipster');

          console.log("tipTooltipRef:", tipTooltipRef);
          console.log("tipTooltipsterContent:", tipTooltipsterContent);


          // update / disable Tooltipster content if necessary
          if (tipTooltipRef == '' || tipTooltipRef == 'undefined'){
            $('.fc-body').find(eID).closest('.tt_tooltip').tooltipster('content', 'Invalid image link ?');
          }


        }
      },

      eventMouseover: function(view, event, element){
        //Not requried for Tooltipster Tooltips - handled by Tooltipster option (trigger: 'hover') 
      },

      eventClick:  function(event, element, view) {
        var e = (event.description);
        if (e != null){
          var chr = e.length;
          // event click coded goes here
          alert(e);
        }
      }

    //end: fullCalendar
    });

    // Even though tooltips are created during [eventRender] the generic class gets initiated here. 
    $('.tt_tooltip').tooltipster(); //{contentCloning: true}

  //end: doc.ready
  });
/* tooltipster.js */
.tooltip_templates {
  display: none;
}
.tooltipster-content{
  /*display: flex;
  flex-direction: column;*/
}
.ttTitle {

}



/*! suit-flex-embed v1.4.0 | MIT License | github.com/suitcss */
.FlexEmbed {
  display: block;
  overflow: hidden;
  position: relative;
}
.FlexEmbed:before {
  content: "";
  display: block;
  width: 100%;
}

.FlexEmbed--16by9:before {
  padding-bottom: 56.25%;
}
.FlexEmbed--4by3:before {
  padding-bottom: 75%;
}
.FlexEmbed--1by1:before {
  padding-bottom: 100%;
}
.CoverImage {
  background-position: 50%;
  background-repeat: no-repeat;
  background-size: cover;
  margin: 0 auto 1em;
  max-height: 600px;
  max-width: 600px;
}
.CoverImageX2 {
  background-color: #808080;
  background-position: 50%;
  background-repeat: no-repeat;
  background-size: 100% 100%;  /*cover; contain;*/
  margin: 0 auto 1em;
  max-height: 2400px;
  max-width: 1200px;
}
<!-- Bootstrap -->
<script src="https://code.jquery.com/jquery-3.3.1.min.js" integrity="sha256-FgpCb/KJQlLNfOu91ta32o/NMZxltwRo8QtmkMRdAu8=" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/popper.js@1.16.0/dist/umd/popper.min.js" integrity="sha384-Q6E9RHvbIyZFJoft+2mJbHaEWldlvI9IOYy5n3zV9zzTtmI3UksdQRVvoxMfooAo" crossorigin="anonymous"></script>
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.4.0/css/bootstrap.min.css" integrity="sha384-SI27wrMjH3ZZ89r4o+fGIJtnzkAnFs3E4qz9DIYioCQ5l9Rd/7UAa8DHcaL8jkWt" crossorigin="anonymous" />
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.4.0/js/bootstrap.min.js" integrity="sha384-3qaqj0lc6sV/qpzrc1N5DC6i1VRn/HyX4qdPaiEFbn54VjQBEU341pvjz7Dv3n6P" crossorigin="anonymous"></script>

<!-- FulCalendar  -->
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fullcalendar/3.10.1/fullcalendar.min.css" integrity="sha256-tXJP+v7nTXzBaEuLtVup1zxWFRV2jyVemY+Ir6/CTQU=" crossorigin="anonymous" />
<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.24.0/moment.min.js" integrity="sha256-4iQZ6BVL4qNKlQ27TExEhBN1HFPvAvAMbFavKKosSWQ=" crossorigin="anonymous"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/fullcalendar/3.10.1/fullcalendar.min.js" integrity="sha256-O04jvi1wzlLxXK6xi8spqNTjX8XuHsEOfaBRbbfUbJI=" crossorigin="anonymous"></script>

<!-- Tooltipster -->
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/css/tooltipster.main.min.css" integrity="sha256-xlmCQ8IjIIx7gqrIAb5x5kEU30jJJm0/DEmrjgLow/E=" crossorigin="anonymous" />
    <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/css/tooltipster.bundle.min.css" integrity="sha256-Qc4lCfqZWYaHF5hgEOFrYzSIX9Rrxk0NPHRac+08QeQ=" crossorigin="anonymous" />
    <script src="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/js/tooltipster.main.min.js" integrity="sha256-9gPC19rdxygnD5cXHFodzczLKeucNZ/dgzLhkKvNtQM=" crossorigin="anonymous"></script>
    <script src="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/js/tooltipster.bundle.min.js" integrity="sha256-NOU7KrY2aTI4PxDegqYUIknk9qfxVCS0E4JfE9aMwaA=" crossorigin="anonymous"></script>
    <script src="https://cdn.jsdelivr.net/npm/tooltipster@4.2.7/dist/js/plugins/tooltipster/SVG/tooltipster-SVG.min.js" integrity="sha256-b9JNfGq08bjI5FVdN3ZhjWBSRsOyF6ucACQwlvgVEU4=" crossorigin="anonymous"></script>

<div id='calendar'></div>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...