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

в 17:45, , рубрики: chrome, Extensions, Google Chrome, javascript, yandex browser, хром, метки:

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

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

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

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

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

В это статье я покажу, как можно создать 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

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

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

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

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

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

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

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

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

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

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

Описание логической структуры расширения как я его вижу до момента написания кода.

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

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

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

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

Вначале это.
Пример создания одного chrome extension - 9

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

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

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

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("Введите название новой папкиrn");
        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("Введите новое название папкиrn" + 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 + "' и все ее содержимое?rnrnВложенные папки и их содержимое удалено не будет.rn");
        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

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

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

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

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

Пример на Github.

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

До ввода теста в строку поиска

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

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

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

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

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

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

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

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

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

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

Основное окно

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

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

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

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

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

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

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

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

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

Автор: Усманов Ринат Ришатович

Источник


https://ajax.googleapis.com/ajax/libs/jquery/3.4.1/jquery.min.js