Интересное поведение mysql при выборке из таблицы содержащей NULL

Столкнулся с интересным поведением MySQL при выборке из таблицы содержащей NULL в качестве значения.  Очередной подвох однако.. Пусть например есть таблица:

mysql> select cat_idx,uuid from categories where tar_id=109 and common=0 and above=1 and archive=0;
+---------+------+
| cat_idx | uuid |
+---------+------+
|       0 | NULL |
|       5 | NULL |
|      10 | NULL |
|      11 | NULL |
|      15 | NULL |
|      28 | peni |
+---------+------+
6 rows in set (0.00 sec)

Хотим получить все значения не содержащие uuid=»peni». Логично было бы использовать такой запрос:

mysql> select cat_idx,uuid from categories where tar_id=109 and common=0 and above=1 and archive=0 and uuid<>"peni";
Empty set (0.00 sec)

Аааа. А куда делись остальные значения? А вот потому что с точки зрения MySQL правильно будет:


mysql> select cat_idx,uuid from categories where tar_id=109 and common=0 and above=1 and archive=0 and (uuid<>"peni" or uuid is null);
+---------+------+
| cat_idx | uuid |
+---------+------+
|       0 | NULL |
|       5 | NULL |
|      10 | NULL |
|      11 | NULL |
|      15 | NULL |
+---------+------+
5 rows in set (0.00 sec)


Вывод: при создании таблиц нужно стараться заполнять поля значениями «по умолчанию». Чтобы NULL не было в принципе. Тогда и возможной ошибки не будет.

 

Наложение выполненных услуг на Яндекс.Карты

Давече сделал интересный скриптик, который иммитирует «онлайн» выполнение заказов и накладывает их на Яндекс.Карты

Демо тут.

<html>
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <link href="https://code.jquery.com/ui/1.11.4/themes/ui-lightness/jquery-ui.css" rel="stylesheet"/>
        <link href="main.css" rel="stylesheet"/>
        <script src="https://code.jquery.com/jquery-3.4.1.min.js"></script>
        <script src="https://code.jquery.com/ui/1.11.4/jquery-ui.min.js"></script>
        <script src="https://api-maps.yandex.ru/2.1/?lang=ru_RU&amp;apikey=df140307-afa8-4232-8513-6308f7fc6f65" type="text/javascript"></script>
</head>
<body>
    <div id="main_layer" style="position: relative; width: 100%;  height: 500px; overflow: hidden;">
        <div class="tickets-wrapper with-list">
            <div id="tickets-map"></div>
            <div id="tickets-list" class="tickets-list">                
            </div>
        </div>
    </div>
    <script>
        ymaps.ready(function () {
            work_show=new Array();
            work_show.push({
                            address : "ул. Герцена 60",
                            nameuser : "Вася",
                            work : "Роутер",
                            summ : 1000,
                            status : "Сделано",
                            img: "waterheater.jpg",                
                            comment: "Абра швабра кадабра!",
                            
                            point : [59.220392, 39.891568], //координаты точки                            
                            ico_preset :  'islands#darkGreenStretchyIcon', // иконка сообщения
                        }
                    );
            work_show.push({
                            address : "ул. Герцена 54",
                            nameuser : "Павел",
                            work : "Газовый котел",
                            summ : 1600,
                            status : "Сделано",
                            img: "waterheater.jpg",
                            comment: "Сделали быстро.Чаевых не просили.",
                
                            point : [59.230492, 39.891568], //координаты точки                            
                            ico_preset :  'islands#darkGreenStretchyIcon', // иконка сообщения
                        }
                    );
            work_show.push({
                            address : "ул. Ленина 54",
                            nameuser : "Вася",
                            work : "Котел",
                            summ : 900,
                            status : "Делаем",
                            img: "waterheater.jpg",
                            comment: "Потек конденцатор. Заменили.",
                
                            point : [59.232492, 39.891568], //координаты точки                            
                            ico_preset :  'islands#darkGreenStretchyIcon', // иконка сообщения
                        }
                    );
            work_show.push({
                            address : "ул. Путина 54",
                            nameuser : "Вася 2",
                            work : "Котел",
                            summ : 100,
                            status : "Делаем",
                            img: "waterheater.jpg",
                            comment: "Потек конденцатор. Заменили.",
                
                            point : [59.232492, 39.893568], //координаты точки                            
                            ico_preset :  'islands#darkGreenStretchyIcon', // иконка сообщения
                        }
                    );
            work_show.push({
                            address : "ул. Карла Маркса 99",
                            nameuser : "Карл-Маркс",
                            work : "Читать книгу",
                            summ : 900,
                            status : "Сделали",
                            img: "waterheater.jpg",
                            comment: "Почитали.Рады.",
                
                            point : [59.237492, 39.891868], //координаты точки                            
                            ico_preset :  'islands#darkGreenStretchyIcon', // иконка сообщения
                        }
                    );
            

        
        
            myMap = new ymaps.Map('tickets-map', {
                    center: [59.220492, 39.891568],
                    zoom: 20,
                    controls: ['rulerControl'],
                   // behaviors: ['default', 'scrollZoom']
                }, {
                    //searchControlProvider: 'yandex#search'
                }),
                    clusterer = new ymaps.Clusterer({

                    preset: 'islands#invertedVioletClusterIcons',
                    groupByCoordinates: false,

                    clusterDisableClickZoom: false,
                    clusterHideIconOnBalloonOpen: false,
                    geoObjectHideIconOnBalloonOpen: false
                }),
                myMap.controls.add(new ymaps.control.ZoomControl({options: { position: { right: 10, top: 50 }}}));
                getPointData = function (index) {
                 return {
                        balloonContentHeader: 'Задача: '+work_show[index].work,
                        balloonContentBody: 'Адрес: '+work_show[index].address+","+work_show[index].nameuser+"<br/>Сумма работ:"+work_show[index].summ+"руб.",
                        balloonContentFooter: work_show[index].comment,
                        clusterCaption: "Сделано: "+work_show[index].summ+"руб.",
                        iconContent: "Сделано: "+work_show[index].summ+"руб.",
                    };
                },
                geoObjects = [];
                messages=new Array();
                for(var i = 0, len = work_show.length; i < len; i++) {
                    geoObjects[i] = new ymaps.Placemark(work_show[i].point, getPointData(i), {preset: work_show[i].ico_preset});                                
                    messages.push('<div id="ticket_id_'+i+'" class="ticket ticket-completed" style="display:none">'+
                            '<div class="ticket-icon" style="background-image: url(images/'+work_show[i].img+')">'+
                            '</div>'+
                            '<div class="ticket-content">'+
                            '    <div class="ticket-title">'+
                            '        <span class="ticket-address">'+work_show[i].address+'</span>'+
                            '        <span class="ticket-contact">'+work_show[i].nameuser+'</span>'+
                            '    </div>'+
                            '    <div class="ticket-category">'+work_show[i].work+'</div>'+
                            '    <div class="ticket-sum">'+work_show[i].summ+'₽</div>'+
                            '    <span class="ticket-state ticket-state-completed">'+work_show[i].status+'</span>'+
                            '</div>'+
                        '</div>');                    
                    if (i<=3) {
                        $("#tickets-list").prepend(messages[i]);
                        $("#ticket_id_"+i).show();
                    };
                }
            clusterer.options.set({
                gridSize: 80,
                clusterDisableClickZoom: true
            });

            clusterer.add(geoObjects);
            myMap.geoObjects.add(clusterer);

            myMap.setBounds(clusterer.getBounds(), {
              //  checkZoomRange: true
            });
            //setTimeout('myMap.setZoom(20);', 2000)            
            // запускаем таймер во время которго перебираем все точки раз в секунду (по кругу)..
             timerId = setInterval(AnimatePoint, 2000);
             gcnt=3; // начинаем с 3-й точки
             function AnimatePoint(){
                 gcnt++;          
                 if (gcnt==work_show.length){gcnt=0;};                 
                 $("#tickets-list").prepend(messages[gcnt]);          
                 $("#ticket_id_"+gcnt).show(1000);
                 myMap.panTo(work_show[gcnt].point, {delay: 500});
                // geoObjects[gcnt].balloon.open();
                // objectState = clusterer.getObjectState(geoObjects[gcnt]);
                 //clusterer.balloon.open(objectState.cluster);
                var objectState = clusterer.getObjectState(geoObjects[gcnt]);
                    if (objectState.isClustered) {
                        objectState.cluster.state.set('activeObject', geoObjects[gcnt]);
                        clusterer.balloon.open(objectState.cluster);
                    } else if (objectState.isShown) {
                        geoObjects[gcnt].balloon.open();
                    }                 
             };            
            
        });

    </script>    
</body>
</html>

Копирование в буфер обмена на JavaScript

В JavaScript скопировать в буфер обмена можно только то что находится внутри тега <input>. Потому приходиться для копирования в буфер обмена произвольного текста приходиться идти на ухищрения в виде например временного помещения текста для копирования в input. Например как-то так:

<span style='cursor: pointer;' id='ipsegsw'>192.168.1.1</span>
<script>
$('#ipsegsw').click(function() {
	    var $temp = $("<input>");
	    $("body").append($temp);
	    $temp.val($('#ipsegsw').text()).select();
	    document.execCommand("copy");
	    $temp.remove();
            console.log('IP скопирован в буфер обмена');
});
</script>