javascript

Я пиарюсь: Пример создания одного extension chrome

  • понедельник, 27 марта 2017 г. в 03:13:16
https://habrahabr.ru/post/324862/
  • JavaScript
  • Google Chrome


Приветствую социум!

Проработал 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. Основная задача топика. Написать быстро расширение без использования фреймворков.