Пример создания одного chrome extension

    Приветствую социум! Проработал 7 лет техническим директором. Понял, насколько это сильно бьет по нервам и решил начать жизнь с чистого листа. Пойти javascript-разработчиком.

    Почему: потому что люблю писать на этом языке. Он веселый и может влет решать множество реальных задач. Да, да, веселый, ведь он настолько не типизирован, что объект может иметь свойство в виде самого себя ) Причем при обращении к этому свойству мы реально изменяем сам объект.

    Веселый JS
    primer = {};
    primer["svoistvo1"] = "reddis";
    primer["svoistvo2"] = primer;
    primer["svoistvo2"]["svoistvo2"]["svoistvo2"]["svoistvo2"]["svoistvo1"] = "dadada";
    console.log(primer);
    



    Конечно без серверной части тяжеловато. Но современные браузеры легко позволяют хранить данные на клиенте. Для решения узконаправленных задач это самое то.

    В это статье я покажу, как можно создать chrome extension.

    Так первым делом создаем пустой проект.

    Как создать пустой проект 'Расширение хром'
    1. Вариант.
    Github
    2. Вариант.
    Ручками.

    Создаем такую структуру.

    • content_scripts (Папка)
      • end.js(Файл)
    • images (Папка)
      • empty_16.png(Файл)
      • empty_32.png(Файл)
      • empty_48.png(Файл)
      • empty_64.png(Файл)
      • empty_128.png(Файл)
    • popup (Папка)
      • popup.html(Файл)
      • popup.js(Файл)
    • manifest.json (Файл)


    Пояснение по структуре.
    content_scripts — Папка где хранятся файлы которые будут добавляться к телу страницы открытой в браузере.
    images — Папка где хранятся иконки расширения
    popup — Папка где хранятся само расширение которое отображается при нажатии на иконку.

    Загрузим наше пустое расширение в браузер.

    Как загрузить пустое расширение в браузер
    Для этого переходим на страницу «chrome://extensions/».

    image

    Выбираем «Режим разработчика».

    image

    Нажимаем кнопку «Загрузить распакованное расширение».



    Выбираем папку где храниться расширение и нажимаем кнопку «OK».



    Если расширение создано правильно, то мы видим его в списке расширений.



    В итоге мы получаем пустое расширение которое работает на всех адресах браузера.

    Так, теперь о идее практической пользы будущего расширения. Ну не знаю может будет транслировать gif-ку где Шэрон Стоун, перекидывает ноги? Конечно потребитель расширения найдется, но целевая группа будет не большой…

    Пример этого расширения

    Не долго думая решил реализовать что то похожее на «Заметки». Но так чтобы и времени немного ушло и расширение на веб приложение было похоже.

    Описание логической структуры расширения как я его вижу до момента написания кода.
    1. Создание, редактирование и удаление заметок.
    2. Создание, редактирование и удаление категорий заметок.
    3. Поиск заметок по описанию и по самим заметкам.

    C чего начну. Рисую прототип. Здесь.

    Я не дизайнер, так что сильно не заморачиваюсь. Просто кидаю нужные мне элементы на экран.

    То что у меня получилось...
    Вначале это.


    А потом вспомнил что:
    Max Width: 792 pixels
    Max Height: 584 pixels.



    Для того, чтобы категории заметок отображались красиво, создал класс и поигрался стилями.

    JS Класс возвращающий DIV с возможностью редактирования
    'use strict';
    
    class Folders {
        constructor( 
        isReturn,
        folders = [
        ["Мои заметки",0],     
        ["Музыка",0],       
        ["Видео",0],    
        ["Документы",0], 
        ["Изображения",0],   
        ["Сайты",0],    
        ["Прочее",0],    
        ],
        saveFolders = console.log,
        selectFunc = console.log,
        deleteFunc = console.log
        ) { 
            if (!isReturn) 
                return;    
            this.selectFunc = selectFunc;
            this.deleteFunc = deleteFunc;
            this.selected = false;
            this.folders = folders;
            this.saveFolders = saveFolders;      
            this.div = document.createElement("DIV");
            this.div.links = this;
            this.div.className = "folderDivVT";
            this.divMenu = document.createElement("DIV");
            this.divMenu.className = "folderMenuVT";
            this.divMenu.innerHTML = "<div class='renameFolder' title='Переименовать папку (F2)'  tabindex='1'></div><div class='deleteFolder' title='Удалить папку (Del)'  tabindex='1'></div><div class='addFolder' title='Добавить папку (Ins)'  tabindex='1'></div><div class='expandFolder' title='Раскрыть все (+)'  tabindex='1'></div><div class='foldFolder' title='Свернуть все (-)'  tabindex='1'></div>";
            this.divMenu.links = this; 
            this.result = document.createElement("DIV");
            this.result.appendChild(this.divMenu);
            this.result.appendChild(this.div);
            this.div.addEventListener("click",function (e) {
                try {
                    this.links.clickP(this.querySelector("p:focus"));
                } catch (ex) {}
            });
            this.div.addEventListener("dblclick",function (e) {    
                try {
                    this.links.clickP(this.querySelector("p:focus"));
                    this.querySelector("p:focus").parentNode.classList.toggle("active"); 
                } catch (ex) {}
            });
            this.div.addEventListener("keydown",function (e) { 
                var parentN = this.querySelector("p:focus").parentNode; 
                var th = false;
                var divs = this.querySelectorAll("DIV");
                for (var i = 0; i < divs.length; i++) 
                        if (divs[i] == parentN)
                            th = i;  
                var rep = -1; 
                var objBounding = false;  
                switch(true) { 
                    case (e.keyCode == 40) && (e.which == 40): 
                            for (var i = th + 1; i < divs.length; i++) {
                                objBounding = divs[i].getBoundingClientRect();
                                if (rep == -1)
                                    if ((objBounding["top"] == 0) && (objBounding["bottom"] == 0) && (objBounding["left"] == 0) && (objBounding["right"] == 0) && (objBounding["width"] == 0)) 
                                        rep = -1;   
                                    else
                                        rep = i;
                            } 
                            if (rep == -1)
                                for (var i = 0; i < th; i++) {
                                    objBounding = divs[i].getBoundingClientRect();
                                    if (rep == -1) {
                                        if ((objBounding["top"] == 0) && (objBounding["bottom"] == 0) && (objBounding["left"] == 0) && (objBounding["right"] == 0) && (objBounding["width"] == 0)) 
                                            rep = -1;   
                                        else
                                            rep = i;
                                    }  
                            }
                            this.links.clickP(divs[rep].querySelector("p"));  
                        break;  
                    case (e.keyCode == 38) && (e.which == 38): 
                            for (var i = th - 1; i > -1; i--) {
                                objBounding = divs[i].getBoundingClientRect();
                                if (rep == -1)
                                    if ((objBounding["top"] == 0) && (objBounding["bottom"] == 0) && (objBounding["left"] == 0) && (objBounding["right"] == 0) && (objBounding["width"] == 0)) 
                                        rep = -1;   
                                    else
                                        rep = i;
                            } 
                            if (rep == -1)
                                for (var i = divs.length - 1; i > th - 1; i--) {
                                    objBounding = divs[i].getBoundingClientRect();
                                    if (rep == -1) {
                                        if ((objBounding["top"] == 0) && (objBounding["bottom"] == 0) && (objBounding["left"] == 0) && (objBounding["right"] == 0) && (objBounding["width"] == 0)) 
                                            rep = -1;   
                                        else
                                            rep = i;
                                    }    
                                }                
                            this.links.clickP(divs[rep].querySelector("p"));
                        break;      
                    case (e.keyCode == 39) && (e.which == 39): 
                            if (!parentN.classList.contains("active"))
                                parentN.classList.add("active"); 
                        break; 
                    case (e.keyCode == 37) && (e.which == 37): 
                            if (parentN.classList.contains("active"))
                                parentN.classList.remove("active");
                        break;
                    case (e.keyCode == 113) && (e.which == 113): 
                            this.links.renameFolder(this.links);
                            this.links.contextDiv.style.display = "none";
                        break;   
                    case (e.keyCode == 46) && (e.which == 46):      
                                this.links.deleteFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                        break;  
                    case (e.keyCode == 45) && (e.which == 45):      
                            this.links.addNewFolder(this.links);
                            this.links.contextDiv.style.display = "none";
                        break; 
                     
                    case (e.keyCode == 107) && (e.which == 107): 
                        var divs = this.links.div.querySelectorAll("DIV.folder");
                        for (var i = 0; i < divs.length; i++) 
                            if (!divs[i].classList.contains("active"))
                                divs[i].classList.add("active");    
                            this.links.contextDiv.style.display = "none";  
                        break;
                    case (e.keyCode == 109) && (e.which == 109): 
                        var divs = this.links.div.querySelectorAll("DIV.folder:not(:first-child)");
                            for (var i = 0; i < divs.length; i++) 
                                if (divs[i].classList.contains("active"))
                                    divs[i].classList.remove("active");  
                            this.links.div.querySelector("DIV.folder:first-child > p").focus(); 
                            this.links.contextDiv.style.display = "none";     
                        break;    
                }           
            });
            this.div.addEventListener("contextmenu",function (e) {   
                try { 
                    var p = this.querySelector("p:focus");
                    this.links.clickP(p);
                    this.links.contextDiv.style.display = "block"; 
                    this.links.contextDiv.style.left = (p.getBoundingClientRect()["left"]+ 50) + "px"; 
                    this.links.contextDiv.style.top = (p.getBoundingClientRect()["top"]+ 10) + "px";
                    e.returnValue = false; 
                } catch (ex) {}
            });
            this.contextDiv = document.createElement("DIV");
            this.contextDiv.className = "folderContext";          
            this.contextDiv.links = this;
            this.contextDiv.style.display = "none";    
            this.contextDiv.innerHTML = '<div class="addFolder"  tabindex="1">Ins Создать подпапку</div><div class="renameFolder"  tabindex="1">F2  Переименовать</div><div class="deleteFolder"  tabindex="1">DEL Удалить</div><div class="Cancel"  tabindex="1">    Отмена</div>';
            this.contextDiv.addEventListener("click",function () {                
                    switch (this.querySelector("DIV:focus").className) {
                        case "addFolder":      
                                this.links.addNewFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                            break;       
                        case "renameFolder":      
                                this.links.renameFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                            break;  
                        case "deleteFolder":      
                                this.links.deleteFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                            break;       
                        case "Cancel": 
                                this.style.display = "none";
                            break;   
                    }
            });
            this.divMenu.addEventListener("click",function () {
                    switch (this.querySelector("DIV:focus").className) {
                        case "addFolder":      
                                this.links.addNewFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                            break;    
                        case "renameFolder":      
                                this.links.renameFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                            break;    
                        case "deleteFolder":      
                                this.links.deleteFolder(this.links);
                                this.links.contextDiv.style.display = "none";
                            break;   
                        case "expandFolder": 
                            var divs = this.links.div.querySelectorAll("DIV.folder");
                            for (var i = 0; i < divs.length; i++) 
                                if (!divs[i].classList.contains("active"))
                                    divs[i].classList.add("active");    
                                this.links.contextDiv.style.display = "none";  
                            break;
                        case "foldFolder": 
                            var divs = this.links.div.querySelectorAll("DIV.folder:not(:first-child)");
                                for (var i = 0; i < divs.length; i++) 
                                    if (divs[i].classList.contains("active"))
                                        divs[i].classList.remove("active");  
                                this.links.div.querySelector("DIV.folder:first-child > p").focus(); 
                                this.links.contextDiv.style.display = "none";     
                            break;
                    }
            });
            document.body.appendChild(this.contextDiv); 
            this.createFolders();
        }
        selectP(p) {
            var selecteds = this.div.querySelectorAll("p.selected");
            for (var i = selecteds.length - 1; i > -1; i--)
                selecteds[i].classList.remove("selected");
            p.classList.add("selected");     
            this.selectFunc(this.div.querySelectorAll("p.selected").dataset.id);
        }
        clickP(p) {
            p.focus();       
            this.selectP(p);
            this.selected = p;                           
            this.contextDiv.style.display = "none"; 
        }
        createFolders() {
            this.div.innerHTML = "";
            for (var i = 0; i < this.folders.length; i++) {
                try {
                    var div = document.createElement("DIV");
                    div.className = "folder";
                    div.id = "folder" + i;
                    div.dataset.id = i;
                    div.dataset.parent = this.folders[i][1];
                    div.innerHTML = "<p tabindex='1'>" + this.folders[i][0] + "</p>";
                    div.querySelector("P").addEventListener("focus",function () {
                        this.click();
                    }); 
                    this.div.appendChild(div);
                } catch (ex) {}    
            }
            
            for (var i = 0; i < this.folders.length; i++) {
                try {
                    this.div.querySelector("#folder" + this.folders[i][1]).appendChild(this.div.querySelector("#folder" + i));
                } catch (ex) {}    
            } 
            try {
                this.clickP(this.div.querySelector("DIV.folder:first-child > p"));   
            } catch (ex) {}    
        }
        addNewFolder(links) {
            var newFolder = prompt("Введите название новой папки\r\n");
            if ((newFolder != "") && (newFolder != null)) {
                var ln = links.folders.length    
                links.folders[ln] = [newFolder,links.selected.parentNode.dataset.id]; 
                var div = document.createElement("DIV");
                div.className = "folder";
                div.id = "folder" + ln;
                div.dataset.id = ln;
                div.dataset.parent = links.selected.parentNode.dataset.id;
                div.innerHTML = "<p tabindex='1'>" + newFolder + "</p>";
                links.div.querySelector("#folder" + links.selected.parentNode.dataset.id).appendChild(div);                                  
                links.saveFolders(links.folders);
                if (!links.div.querySelector("#folder" + links.selected.parentNode.dataset.id).classList.contains("active"))
                    links.div.querySelector("#folder" + links.selected.parentNode.dataset.id).classList.add("active");
                links.clickP(div.querySelector("p"));
            }
        }
        renameFolder(links){
            var newFolder = prompt("Введите новое название папки\r\n" + links.selected.innerHTML,links.selected.innerHTML);  
            if ((newFolder != "") && (newFolder != null)) {
                links.selected.innerHTML = newFolder;
                links.folders[links.selected.parentNode.dataset.id][0] = newFolder;                     
                links.saveFolders(links.folders);
            }
        }
        deleteFolder(links){                         
            var id = links.selected.parentNode.dataset.id;
            if (id == 0) {
                alert("Нельзя удалять главную папку");
                return;
            }
            var delFolder = confirm("Вы точно хотите удалать папку '" + links.selected.innerHTML + "' и все ее содержимое?\r\n\r\nВложенные папки и их содержимое удалено не будет.\r\n");
            if (delFolder) {
                var parendDiv = links.selected.parentNode.parentNode.id;
                links.folders[links.selected.parentNode.dataset.id] = null;     
                links.createFolders();  
                var div = links.div.querySelector("#" + parendDiv);
                div.classList.add("active");
                while (true) {
                    div = div.parentNode;             
                    if (div.classList.contains("folderDivVT"))
                        break;   
                    div.classList.add("active");
                }
                links.clickP(links.div.querySelector("#" + parendDiv + " > P"));
                links.deleteFunc(id);   
                links.saveFolders(links.folders);         
            }        
        }
    }
    


    Стили для класса
    .folderDivVT div > p.selected {
        background:#4DB6AC;                     
        color:#f5f5f5;
    }       
    
    .folderDivVT div > p.selected:focus,.folderDivVT div > p:focus {
        background:Teal;
        color:#fff;
    }    
    p:first-letter {
        //color:Teal;
    }
    
    .folderDivVT div {
        margin:5px;
        margin-left:15px;
        width:100%;
    }
    
    .folderDivVT div > p {
        margin:0;
        cursor:pointer;
        padding:3px;
        display:inline-block;   
        white-space: normal;
        word-break: break-word;
        max-width:calc(100% - 22px);
    }
    
    .folderDivVT div > div {
        display:none;
    } 
    
    .folderDivVT div.active > div {
        display:block;
    }
    
    #foldersDiv {
        position:fixed;
        left:0;
        top:50px;
        padding:10px;
        width:300px;
        overflow:auto;
        height:calc(100% - 50px);
        z-Index:1000;
    } 
    
    .folderDivVT div:before {  
        content: " ";  
        color: #fff;  
        background-image: url('');    
        display:inline-block;
        width: 16px;
        height: 16px;
        background-size: cover;
        margin-right:5px;       
        float: left;
      
    }
    
    .folderDivVT div.active:before {  
        background-image: url('');    
    }
    .folderDivVT {
        overflow: auto;
        height: calc(100% - 28px);
    }
    
    .folderMenuVT {  
        text-align:center;
        width:100%;          
        border-bottom:1px solid #999;
    }
    .folderMenuVT > div:before {  
        content: "";  
        color: #fff;  
        display:inline-block;
        width: 24px;
        height: 24px;
        background-size: cover;
        margin:0;
        margin-right:5px;  
        cursor:pointer;
      
    }
    .folderMenuVT > div {  
        display:inline-block;
    }
    
    .replaceFolder:before {
        background-image:url('');
    }                     
    .renameFolder:before {
        background-image:url('');
    }                     
    .deleteFolder:before {
        background-image:url('');
    }                    
    .addFolder:before {
        background-image:url('');
    }
    .expandFolder:before {
        background-image:url('');
    }
    .foldFolder:before {
        background-image:url('');
    }
                     
    
    #folderMainDiv {
        width:300px;
        height:100%;
        position:fixed;
        top:0;
        left:0;
        border-right:1px solid #999;    
        padding-top: 10px;
        background-color:rgba(255,255,255,1);
    }
    
    
    .folderContext {
        position:fixed;
        border:2px ridge #004D40;
        position:fixed;
        width:180px;     
        padding: 0px;
        cursor: pointer;
        color:#050505;  
        background: #fff;
        z-Index:1001;
    }
    .folderContext > div {
        padding:5px;          
        border:1px solid rgba(255,255,255,0);  
        border-width: 1px 0 1px 0;
    }             
    .folderContext > div:hover {
        background-color: #00695C; 
        color:#ffffff;  
        border:1px solid #26A69A;
        border-width: 1px 0 1px 0;
    }
    .folderContext > div:first-Child:hover {
        border-top:1px solid #00695C;     
    }
    .folderContext > div:last-Child:hover {
        border-bottom:1px solid #00695C;     
    }
    .folderContext > div:before {  
        content: "";  
        color: #fff;  
        display:inline-block;
        width: 16px;
        height: 16px;
        background-size: cover;
        margin:0;
        margin-right:5px;  
      
    }
    
    #tempContextFolder {
        display:none;
    }
    


    Пример использования класса можно увидеть на Github.

    Кому лень ходить на Github


    В итоге получился класс, в котором работают кнопки верх, низ, влево, вправо, таб, ins, del, f2, а также естественно мышь, и контекстменю ))).

    Папки с блекджеком и…

    Следующим делом добавляю в строке поиск «event» на событие «input», для того чтобы результаты поиска появлялись моментально.

    Пример на Github.

    Для тех кому лень экспериментировать
    До ввода теста в строку поиска



    После того как пользователь начал вводить данные в поисковую строку



    То же только в DIV добавлено css свойство opacity 0.5




    Ну и в конце редактирую manifest.json. И даю название расширению «Notes beta 1».

    Редактирование названия



    Расширение готово к использованию.

    Фото отчет --- Как оно работает
    Основное окно



    Создание новой заметки



    Отображение заметки при наведении мышки на нее



    Поиск заметок



    Код расширения на Github.

    p.s. Жду конструктивной критики.
    p.p.s. В классе Folders не работает функция «Переместить». На реализацию логики немного не хватило времени. Также все картинки в стилях преобразованы в base64.
    p.p.p.s. Основная задача топика — написать быстро расширение без использования фреймворков.
    Share post

    Comments 16

      +7
      Ну, коль уж вы хотели конструктивной критики:
      — не стоит вызывать getBoundingClientRect() для одного и того же элемента подряд строка за строкой — это добавляет нагрузку, лучше вынести в отдельную переменную первый запуск, а затем уже обращаться к свойствам;
      — обращаться к свойствам объекта можно не только через квадратные скобки, но и через точку;
      — document.getElementById('title') сработает быстрее, чем document.querySelector("#title") (так для всех обращений по id) (про window.title молчу, т.к. лучше про него забыть));
      — для элементов, которые не меняются и к которым часто происходит обращение, лучше завести отдельный объект типа кэша, чтобы каждый раз через селекторы не обращаться (или в вашем случае можно «забиндить» слушателей событий на this, чтобы обращаться к тем же самым созданным div'ам);
      — стили лучше менять с помощью классов — браузер в таком случае применяет сразу весь набор новых стилей, а не строчка за строчкой, каждый раз перерисовывая элемент;
      — код лучше разнести по разным файлам, в зависимости от назначения;
      — стили так же лучше вынести туда, где они должны быть, то есть в css-файлы;
      — если появление дополнительных ненужных отступов заставляет вас писать в строку элементы html, то можно убрать шрифт для всей страницы (font-size:0px;), а включить его там, где он непосредственно нужен, чтобы красивее отобразить html-код;
      — честно говоря, не понимаю, для чего вам end.js, лучше его удалить, а так же убрать из манифеста возможность редактировать страницу любого сайта — это вызовет больше доверия у потенциальных пользователей;
      — структура рода [«Мои записки»,0] ведет к ненужной жесткости кода — вам потом понадобиться добавить еще какое-то свойство, придется расширять массив и менять функцию-обработчик, я бы советовал заменить на объект;
      — очень много дублирования кода;
      — на каждое нажатие клавиши вы запускаете перебор по всем div'ам — это ужасно — если вы хотите найти следующий (предыдущий) элемент за родителем абзаца, который находится в фокусе, пользуйтесь nextElementSibling /previousElementSibling, с проверкой на существование;
      — вы сравниваете e.keyCode и e.which на одно и то же значение, используйте любой из них (лучше keyCode) и пользуйтесь своим switch по-нормальному на здоровье (если вдруг так уж случилось, что они у вас работают по-разному, то можно воспользоваться конструкциями вида key=e.keyCode||e.which);
      — вместо одного универсального слушателя лучше вешать на каждый элемент свой и отдельно повесить на нажатие по контекстному меню на скрытие;
      — и к детям лучше все-таки обращаться через children (childNodes).
      Честно скажу, код особо не читал, написал только то, что сразу бросилось в глаза, поэтому есть ли в логике самого расширения какие-то ошибки и недочеты, не могу сказать (код отвлекает). Для чего используется focus на элементах p тоже не знаю, но, думаю, лучше обойтись без него.
      Основным итогом будет следующий: не используйте querySelector как панацею, ищите более быстрые альтернативы, применяйте файлы html, js и css для тех целей, для которых они предназначаются, в манифесте должно быть перечислено только то, что используется (файлы, разрешения).
        +1
        Спасибо большое за комментарий.

        — не стоит вызывать getBoundingClientRect() для одного и того же элемента подряд строка за строкой — это добавляет нагрузку, лучше вынести в отдельную переменную первый запуск, а затем уже обращаться к свойствам;
        В классе я так и действовал, нашел лишь одно место где обращался дважды к getBoundingClientRect().

        — обращаться к свойствам объекта можно не только через квадратные скобки, но и через точку;
        Спасибо, что напомнили. Первый мой язык для веба был php. Он показывал ошибку при обращении в stdClass() к элементу имеющему имя на кириллице.Это привело к тому, что я привык в php обращаться к свойствам через скобки. И сказалось на мой стиль программирования в JS.

        Как сейчас состоит дело со свойствами на кирилице







        — document.getElementById('title') сработает быстрее, чем document.querySelector("#title") (так для всех обращений по id) (про window.title молчу, т.к. лучше про него забыть));
        Я отказался работать с getElementById() в тот момент когда понял что он выдергивает только одно значение. В теле страницы могут быть несколько элементов с одним id, при парсинге товарных агрегаторов я раз и навсегда отказался от getElementById(), и перешел на querySelector() и querySelectorAll()

        — для элементов, которые не меняются и к которым часто происходит обращение, лучше завести отдельный объект типа кэша, чтобы каждый раз через селекторы не обращаться (или в вашем случае можно «забиндить» слушателей событий на this, чтобы обращаться к тем же самым созданным div'ам);
        Я не совсем понял о чем тут речь.

        — стили лучше менять с помощью классов — браузер в таком случае применяет сразу весь набор новых стилей, а не строчка за строчкой, каждый раз перерисовывая элемент;
        Так я итак использую классы. Не везде конечно, но там где мне нужно точно переопределить css к элементу. Так как обращение в ccs к id(#) важнее чем обращение к классу элемента(.)

        — код лучше разнести по разным файлам, в зависимости от назначения;
        — стили так же лучше вынести туда, где они должны быть, то есть в css-файлы;
        — честно говоря, не понимаю, для чего вам end.js, лучше его удалить, а так же убрать из манифеста возможность редактировать страницу любого сайта — это вызовет больше доверия у потенциальных пользователей;
        — структура рода [«Мои записки»,0] ведет к ненужной жесткости кода — вам потом понадобиться добавить еще какое-то свойство, придется расширять массив и менять функцию-обработчик, я бы советовал заменить на объект;
        — очень много дублирования кода;
        — вы сравниваете e.keyCode и e.which на одно и то же значение, используйте любой из них (лучше keyCode) и пользуйтесь своим switch по-нормальному на здоровье (если вдруг так уж случилось, что они у вас работают по-разному, то можно воспользоваться конструкциями вида key=e.keyCode||e.which);
        p.p.p.s. Основная задача топика — написать быстро расширение без использования фреймворков. Смысл быстро написать и запустить

        — если появление дополнительных ненужных отступов заставляет вас писать в строку элементы html, то можно убрать шрифт для всей страницы (font-size:0px;), а включить его там, где он непосредственно нужен, чтобы красивее отобразить html-код;
        За этот трюк большое спасибо!!!

        — на каждое нажатие клавиши вы запускаете перебор по всем div'ам — это ужасно — если вы хотите найти следующий (предыдущий) элемент за родителем абзаца, который находится в фокусе, пользуйтесь nextElementSibling /previousElementSibling, с проверкой на существование;
        nextElementSibling /previousElementSibling взял на заметку. Просто забыл про существование это навигации.

        — вместо одного универсального слушателя лучше вешать на каждый элемент свой и отдельно повесить на нажатие по контекстному меню на скрытие;
        Вот здесь Вы не правы. Я создал массив папок из 200 элементов. Потом повесил на каждый элемент по обработчику события. В итоге Страница с действующим классом, начала подвисать при event'ах, а у меня очень даже нормальный ноутбук. И учитывая, что на странице может быть не один экземпляр класса мне пришлось переписать логику так чтобы event'ов было не так много.

        и еще раз большое спасибо за такой развернутый комментарий.
          +2
          Я отказался работать с getElementById() в тот момент когда понял что он выдергивает только одно значение. В теле страницы могут быть несколько элементов с одним id

          Могут, но не должны, id есть уникальный идентификатор элемента, наличие нескольких — ошибка семантики. Для выбора нескольких элементов есть более другие методы, содержащие elements в названии (getelementsbyclassname() например).
            0
            Полностью с Вами согласен.

            Но закон не писан для многих.

            Да и в принципе это не создает никаких проблем если обращаться к id посредством querySelector().
            0
            В классе я так и действовал, нашел лишь одно место где обращался дважды к getBoundingClientRect().

            Я имел в виду:
            this.links.contextDiv.style.left = (p.getBoundingClientRect()[«left»]+ 50) + «px»;
            this.links.contextDiv.style.top = (p.getBoundingClientRect()[«top»]+ 10) + «px»;
            и
            var leftCR = a.getBoundingClientRect()[«left»];
            var topCR = a.getBoundingClientRect()[«top»];
            var widthCR = a.getBoundingClientRect()[«width»];
            var bottomCR = a.getBoundingClientRect()[«bottom»];


            Я отказался работать с getElementById() в тот момент когда понял что он выдергивает только одно значение. В теле страницы могут быть несколько элементов с одним id, при парсинге товарных агрегаторов я раз и навсегда отказался от getElementById(), и перешел на querySelector() и querySelectorAll()
            Просто оставлю это здесь. Но, в конце концов, это ваш продукт, поэтому вам принимать окончательное решение в плане выбора способа, мы лишь советуем.

            По поводу отдельного объекта для кэша — здесь все зависит от логики работы вашего расширения — но в виду того, что элементы на странице не могут возникать из ниоткуда, а создаются непосредственно js-скриптом, то можно ссылки на них помещать в отдельный объект, чтобы через него быстрее обращаться к элементам, даже посредством того же перебора. Хранение не сильно нагрузит память, т.к. в объекте или массиве будут храниться ссылки на объекты.

            По поводу «биндинга» — Привязка контекста и карринг: «bind».

            Вот здесь Вы не правы. Я создал массив папок из 200 элементов. Потом повесил на каждый элемент по обработчику события. В итоге Страница с действующим классом, начала подвисать при event'ах, а у меня очень даже нормальный ноутбук. И учитывая, что на странице может быть не один экземпляр класса мне пришлось переписать логику так чтобы event'ов было не так много.

            Здесь я имел в виду контекстное меню.
            Напишите, пожалуйста, пример того, как вы вешали слушателей (или дайте ссылку на него) — интересно взглянуть, т.к. у самого есть проект с достаточно большим числом объектов на странице, на каждом из которых висит отдельный слушатель, и пока тормозов и подвисаний замечено не было.
              0
              Я отказался работать с getElementById() в тот момент когда понял что он выдергивает только одно значение. В теле страницы могут быть несколько элементов с одним id, при парсинге товарных агрегаторов я раз и навсегда отказался от getElementById(), и перешел на querySelector() и querySelectorAll()
              Просто оставлю это здесь. Но, в конце концов, это ваш продукт, поэтому вам принимать окончательное решение в плане выбора способа, мы лишь советуем.


              Насчет скорости доступа я согласен. Быстрее обращение к id. Но дело не в скорости, а в том я для себя решил работать в js c css селекторами. Возможно в каком то другом проекте и потребуется использовать именно getElementById, но пока querySelector.

              Тесты






              Плюс по скринам видно, что миллион обращение к id и к селектору примерно выполняются за 100 и 300 секунд. Для меня не критично в данном проекте ускоряться за счет id.

              Вот здесь Вы не правы. Я создал массив папок из 200 элементов. Потом повесил на каждый элемент по обработчику события. В итоге Страница с действующим классом, начала подвисать при event'ах, а у меня очень даже нормальный ноутбук. И учитывая, что на странице может быть не один экземпляр класса мне пришлось переписать логику так чтобы event'ов было не так много.

              Здесь я имел в виду контекстное меню.
              Напишите, пожалуйста, пример того, как вы вешали слушателей (или дайте ссылку на него) — интересно взглянуть, т.к. у самого есть проект с достаточно большим числом объектов на странице, на каждом из которых висит отдельный слушатель, и пока тормозов и подвисаний замечено не было.


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

              И еще раз спасибо за критику.
          +1
          Пример создания одного extension chrome

          тогда уж или «Пример создания одного chrome extension» или «Пример создания одного chrome-расширения» или «Пример создания одного расширения для chrome».
            0
            Спасибо, перечитал несколько раз заголовок. Понял что к чему прилагается )))
            0
            А можно использовать Yeoman и генератор Yeoman.
            • UFO just landed and posted this here
                0
                Да это все влияние Java.

                На досуге начал изучать, и оттуда потянуло.
                0

                Получился не проект, а сильносвязанный блоб.


                1. Класс-портянку рассортировать, ибо SOLID;
                2. Мух от котлет отделить. HTML, CSS и JS должны быть в отдельных файлах;
                3. Менять CSS стили во время выполнения нехорошо и неинформативно.
                4. Соглашения о наименованиях и стиле кода существуют не просто так;
                5. CSS без препроцессора? В 2017 году?

                Взаимодействия с WebSQL или IndexedDB тоже не видно. Это что только интерфейс?

                0
                Да, да, веселый, ведь он настолько не типизирован, что объект может иметь свойство в виде самого себя ) Причем при обращении к этому свойству мы реально изменяем сам объект.

                Вот только типизация тут ни при чём. В типизированных языках такое тоже возможно.
                  +1
                  Вот это
                  $obj->Русские_буквы

                  — зло.
                  Можете объяснить, зачем Вам кириллица в именах?
                  Вот так, что без этого совсем никуда…
                  Если это про «национальную гордость великороссов», то представьте, что вам достался на поддержку китайский проект, а они гордыми оказались…
                  Про
                  «svoistvo»
                  вообще промолчу… Чем «property» не угодило? Английского не знаете?
                  Javascript изучаете, изучите азы английского заодно. Не трудно и пригодится много где еще.

                  Общее впечатление от Вашего подхода — нет структуры, каша какая-то.
                  Уж простите за такую критику.
                    0
                    YuryZakharov, спасибо за комментарий.

                    Вот это
                    $obj->Русские_буквы

                    — зло.
                    Можете объяснить, зачем Вам кириллица в именах?
                    Вот так, что без этого совсем никуда…
                    Если это про «национальную гордость великороссов», то представьте, что вам достался на поддержку китайский проект, а они гордыми оказались…


                    Вот смотрите, допустим вы парсите какой нибудь сайт, в его коде элементы различаются по dataset свойствам (к примеру data-auth=«Тиничев Валерий»). Вот именно при обработке таких блоков и возникают не англоязычные имена свойств.
                    Или другой пример.
                    Вы обрабатываете текст на поиск повторяющихся значений. Намного проще давать key именно часть текста не зависимо от языка. Это лишь при обработке данных.

                    «svoistvo»
                    вообще промолчу… Чем «property» не угодило? Английского не знаете?
                    Javascript изучаете, изучите азы английского заодно. Не трудно и пригодится много где еще.


                    property — всем угодило. Но для быстроты и наглядности использовал именно русское слово на латинице.

                    Общее впечатление от Вашего подхода — нет структуры, каша какая-то.

                    p.p.p.s. Основная задача топика — написать быстро расширение без использования фреймворков. Быстро написать в течении дня, максимум двух. И быстро запустить.

                    Уж простите за такую критику.

                    Наоборот спасибо Вам за критику, если человек не критиковать он остановиться в развитии. Ведь сам себя любой может простить и понять. )))

                  Only users with full accounts can post comments. Log in, please.