Форум программистов, компьютерный форум, киберфорум
JavaScript: API
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.91/11: Рейтинг темы: голосов - 11, средняя оценка - 4.91
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
Chrome Extension

Как исправить работу chrome.tabs.remove?

15.01.2018, 17:39. Показов 2220. Ответов 18
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Доброго времени суток. Имеется расширение, которое было допилинно под свои нужды. Расширение умеет закрывать вкладки, если их более заданного числа, после чего, нажимает на нужную кнопку на странице. Проблема состоит в том, что код закрывает не только вкладки, но и новооткрытые окна, будь они открыты из под страницы или вручную. По идее chrome.tabs.remove должен работать только с вкладками и даже не знаю как исправить его, что бы не трогал окна.

JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
//background.js
var isEnabled = true;
var maxTabs = 1;
var tabsCount;
 
 
function updateBadgeText() {
    var tabsBalance = maxTabs - tabsCount;
    var tabsAllowanceRemaining = (tabsBalance > 0) ? tabsBalance : 0;
 
    chrome.browserAction.setBadgeText({
        text: "" + tabsAllowanceRemaining
        
    });
    
}
 
function updateTabsCount() {
    
    chrome.tabs.query({
        windowType: 'normal',
        pinned: false
    }, function (tabs) {
        tabsCount = tabs.length;
        updateBadgeText();
    });
}
 
 
 
function handleTabCreated(tab) {
   if (tabsCount >= maxTabs) {
        setTimeout(function() {
        chrome.tabs.remove(tab.id);
        }, 2000);   
    }
    else {
        updateTabsCount();
    }
}
 
 
 
 
function handleTabRemoved(tab,response)
{   
    updateTabsCount();
    if (response.isWindowClosing == false)
    {
            chrome.tabs.query({ "active": true, "windowId": response.windowId }, function (tabs)
            {
                chrome.tabs.reload(tabs[0].id, function ()
                {
                    began(tabs[0].id);
                
                });
            });
    }
}
    
 function began(tabId) {
    setTimeout( function() {
        chrome.tabs.sendMessage(tabId, {greeting: "hello"}, function(response) 
   {});
   }, 2000);
}
 
function handleTabUpdated(tab) {
    updateTabsCount();
}
 
 
function init() {
    updateTabsCount();
    chrome.tabs.onCreated.addListener(handleTabCreated);
    chrome.tabs.onRemoved.addListener(handleTabRemoved);
    chrome.tabs.onUpdated.addListener(handleTabUpdated);
    }
 
function teardown() {
    chrome.tabs.onCreated.removeListener(handleTabCreated);
    chrome.tabs.onRemoved.removeListener(handleTabRemoved);
    chrome.tabs.onUpdated.removeListener(handleTabUpdated);
    }
 
chrome.browserAction.onClicked.addListener(function (tab) {
    if (!isEnabled) {
        init();
        chrome.browserAction.setIcon({ path: "icons/19.png" });
    }
    else {
        teardown();
        chrome.browserAction.setIcon({ path: "icons/19-disabled.png" });
        chrome.browserAction.setBadgeText({ 'text': '' });
  }
 
    isEnabled = !isEnabled;
});
 
init();
JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
//content.js
function handleTabClick() {
 
                document.getElementById('0').click();
 
}
 
 
chrome.runtime.onMessage.addListener(
  function(request, sender) {
    if (request.greeting == "hello"){
   handleTabClick();
    }
  });
JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
//manifest.js
{
    "name": "Tab Limit",
    "description": "Limit the number of open, non-pinned browser tabs",
    "version": "0.1.0",
    
    "background": {
        "scripts": ["background.js"]
    },
    "content_scripts": [
    {
      "matches": [ "*://*мой_сайт*/*" ],
      "js": [ "content.js" ],
      "run_at": "document_start"
    }
 ],
    "permissions": [
        "tabs"
        
    ],
    "browser_action": {
        "default_icon": {
            "19": "icons/19.png",
            "38": "icons/38.png"
        },
        "default_title": "Tab Limit"
    },
    "icons": {
        "48": "icons/48.png",
        "128": "icons/128.png"
    },
    "manifest_version": 2
}
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
15.01.2018, 17:39
Ответы с готовыми решениями:

Chrome.tabs.executeScript() - ошибка
Добрый день, использую функцию chrome.tabs.executeScript() чтобы встроить код в конкретную указанную страницу, на моем пк все работает...

Chrome.tabs - получить html код вкладки
Добрый день, подскажите пожалуйста с какими функциями chrome.tabs можно получить Html код активной вкладки узнать url адрес этой вкладки? ...

Chrome DevTools. Не отображаются подсказки в javascript коде. Как исправить?
Пишу javascipt код в Google Chrome (Версия 65.0.3325.181). Находится всё в локальном файле. Когда пишу код в консоли, тогда...

18
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 11:54
Цитата Сообщение от BeleK0ss Посмотреть сообщение
По идее chrome.tabs.remove должен работать только с вкладками
Да так-то оно так, только вот если открыть окно с единственной вкладкой и закрыть эту вкладку, то и окно тоже закроется.
Цитата Сообщение от BeleK0ss Посмотреть сообщение
даже не знаю как исправить его, что бы не трогал окна.
Ну, например можно не трогать вкладку, если она единственная в окошке.
Проблема возникает в функции handleTabCreated. Там удаление вкладки происходит если tabsCount превышает maxTabs, но переменную tabsCount надо постоянно обновлять для того, чтобы она содержала актуальное количество вкладок, а так она содержит то значение, которое имело место при последнем обновлении. Вообще, использование общедоступной переменной для таких целей - плохое решение. Можно, конечно, обновить и все такое, но в принципе гарантированно избежать проблем при таком подходе не получится. Вместо этого количество вкладок надо вычислять каждый раз, когда оно понадобится. В крайнем случае вычисление вынести в отдельный метод и вызывать его по мере надобности(не забывая об асинхронности операций, разумеется). Кроме того, обрати внимание на то, что при обновлении этого значения ты не учитываешь многооконности. В твоем запросе вычисляется общее количество вкладок во всех окнах. Из-за этого даже если ты исправишь первую проблему, новые окна все равно будут закрываться в тех случаях, когда суммарное количество вкладок всех этих окон достигнет максимума. Если ты ждешь именно такого поведения, то все нормально, в противном случае - при вычислении количества вкладок указывай какие именно вкладки берутся в расчет.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 12:25  [ТС]
Я понял. А можно как-то отключить действие расширения или скрипта только на 1 сайте или странице? Мне по сути то и нужно что бы при вызове таблицы на одном сайте, которая открывается в новом окне, не закрывалась.
Может как-то можно указать исключение?
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 12:32
BeleK0ss, я не очень понял, что именно ты имеешь в виду, но в принципе, я уже написал, что при подсчете вкладок, надо учитывать, какие именно вкладки берутся в расчет. Что касается включения-отключения скрипта на конкретных страницах, то в принципе тебе никто не мешает перед выполнением действия проверять урл документа, для которого ты это действие выполняешь. Посмотри опции метода tabs.query, там довольно много возможностей, да и полученные вкладки тоже можно отфильтровать. Кроме того, для контент-скриптов можно прописать, где они должны работать. Так что там есть варианты и надо по задаче смотреть, что лучше.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 12:37  [ТС]
Content да, знаю. Но основные функции у меня выполняются ведь в Background...
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 12:47
Цитата Сообщение от BeleK0ss Посмотреть сообщение
Content да, знаю. Но основные функции у меня выполняются ведь в Background...
Ну они же взаимодействуют. Всегда можно из того же бэкграунда запросить какую-то информацию у контента. Кроме того, какую-то часть логики можно в контент перенести.
Я не говорю, что это оптимальный вариант в данном случае, просто описал разные варианты. В твоем случае, по всей видимости, оптимальным будет именно проверка урл вкладки, для которой собираешься выполнить действие.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 12:57  [ТС]
На сколько я понял, нужно указать через query на каких страницах ему работать через опцию url примерно вот так ?
JavaScript
1
2
3
4
5
6
7
8
9
10
11
function updateTabsCount() {
    
    chrome.tabs.query({
        url: "https://www.extension.com/*",
        windowType: 'normal',
        pinned: false
    }, function (tabs) {
        tabsCount = tabs.length;
        updateBadgeText();
    });
}
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 13:11
Цитата Сообщение от BeleK0ss Посмотреть сообщение
На сколько я понял
Проблема в том, что я не все понял. Я уже написал, что там логика может меняться, в зависимости от того, какие вкладки следует учитывать и т. д. Тот запрос, который ты написал, опять-таки будет возвращать вкладки всех окон. Тебе нужно именно это или ограничение на количество вкладок действует для каждого окна отдельно? В данном случае если во всех окнах будет открыто максимальное количество вкладок с данного сайта, то новые окна будут закрываться, точно так же, как это происходит сейчас. Кроме того, я еще раз настоятельно советую отказаться от переменной, хранящей количество вкладок.
Что касается проверки урл, то это был ответ на вопрос о том, как выполнить действие для одних сайтов и не выполнять для других. Ну у тебя же есть объект Tab, полученный из query, в нем есть свойство url, проверь его значение и если это левая вкладка, то ничего не делай. Ну что-то типа
JavaScript
1
2
3
4
if(tabs[0].url.startsWith("https://www.extension.com"))
{
// Здесь делаем что надо
}
А по поводу запросов, ну вот например там можно указать, что запрашиваются вкладки активного окна, просчитал сколько их, и если в окне только одна вкладка - ничего не закрываешь. Что-то более конкретное могу посоветовать, если буду более точно понимать, что именно ты хочешь сделать и как твое расширение должно вести себя в той или иной ситуации.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 15:49  [ТС]
Расширение должно закрывать новооткрытую вкладку. Не обязательно что бы она была только 1 из всех. Главное не допустить открытия новых вкладок.
Так же после закрытия новой вкладки, скрипт должен обновить предыдущую вкладку(chrome.tabs.reload) и нажать на нужную кнопку (document.getElementById('0').click().
Дело в том, что исходник расширения я взял на просторах гитахаба, и подпилил под себя все остальное. И до последнего работало все очень хорошо, но на странице есть таблица которая открывается в новом окне. И вот нужно как-то сделать, что бы эта страница, открытая в новом окне, не закрывалась по запросу скрипта (chrome.tabs.remove(tab.id), что бы она исключалась через свойство url или даже пусть подсчитывает и отклоняет запрос если осталась одна вкладка. Попробовал
JavaScript
1
2
3
4
if(tabs[0].url.startsWith("https://www.extension.com"))
{
// Здесь делаем что надо
}
Пока что не выходит особо.
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 16:41
Цитата Сообщение от BeleK0ss Посмотреть сообщение
Попробовал
Я это написал не для того, чтобы пробовать. Это была примерная иллюстрация того, как надо действовать, когда определенные действия должны выполняться только для некоторых страниц. Здесь осуществляется проверка, что адрес вкладки начинается с заданного текста и только в этом случае будет выполняться действие.

Теперь что касается самой проблемы. Попытаюсь объяснить более подробно, что именно происходит. Когда открывается новое окно, вместе с ним создается и новая вкладка. Далее происходит событие, запускающее функцию handleTabCreated. В этой функции ты проверяешь переменную tabsCount, которая на этот момент содержит количество вкладок другого окна (не вновь созданного) на момент последнего вызова функции, обновляющей эту переменную. И если при последнем обновлении значение этой переменной превышало максимально допустимое количество вкладок, то будет закрываться любая вновь открытая вкладка, независимо от того, где она была открыта. То есть проблема именно в способе получения информации о количестве открытых вкладок. Я уже вверху несколько раз написал, что надо избавиться от переменной tabsCount и функции, обновляющей эту переменную, поскольку кроме путаницы этот ничего не принесет. Тебе нужно подсчитывать либо количество вкладок активного окна, либо, если это делается в обработчике открытия вкладки, можно запросить количество вкладок того же окна, в котором открыта эта вкладка. Конечно, делать каждый раз все эти запросы не очень удобно и было бы лучше вынести их в отдельную функцию. Тут правда возникает другая проблема - поскольку запросы асинхронные, то функция, возвращающая их результат тоже должна быть асинхронной. Могу предложить такой вариант решения для вычисления количества вкладок активного окна.
JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
function getTabsCount()
{
    return new Promise(function (resolve)
    {
        chrome.windows.getCurrent(function (win)
        {
            chrome.tabs.query({ windowId: win.id }, function (tabs)
            {
                resolve(tabs.length);
            });
        });
    });
}
Соответственно обработчик события тоже делаем асинхронным, чтобы функцию можно было вызвать с помощью await
JavaScript
1
2
3
4
5
6
7
8
9
10
11
async function handleTabCreated(tab)
{
    let tabsCount = await getTabsCount();
    if (tabsCount >= maxTabs)
    {
        setTimeout(function ()
        {
            chrome.tabs.remove(tab.id);
        }, 2000);
    }
}
Ну и в других местах, где ты используешь переменную, лучше сделать так же как и здесь, а от переменной избавиться.
Можно функцию вычисления количества вкладок немного улучшить, чтобы она возвращала количество вкладок активного окна, когда не получает аргументов, а если ей передать id окна, то она возвратит количество вкладок в этом окне.
JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
function getTabsCount(winId)
{
    return new Promise(function (resolve)
    {
        chrome.windows.getCurrent(function (win)
        {
            chrome.tabs.query({ windowId: winId ? winId : win.id }, function (tabs)
            {
                resolve(tabs.length);
            });
        });
    });
}
Тогда код обработчика изменится так
JavaScript
1
2
3
4
5
6
7
8
9
10
11
async function handleTabCreated(tab)
{
    let tabsCount = await getTabsCount(tab.windowId);
    if (tabsCount >= maxTabs)
    {
        setTimeout(function ()
        {
            chrome.tabs.remove(tab.id);
        }, 2000);
    }
}
Но все это надо пробовать, поскольку писал без проверки, может и ошибся где.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 17:14  [ТС]
Подставил, везде заменил updateTabsCunt на getTabsCount, но ничего не изменилось. отладчик молчит.
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 17:27
BeleK0ss, не заменить надо, а вызывать функцию вместо использования переменной, от которой вообще надо избавиться. Кроме того надо использовать async...await, без этого работать не будет.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 17:43  [ТС]
Я, наверное, просто не правильно выразился. Сделал как вы сказали, убрал переменную и функцию которая её обновляет. Внес изменения с вашими правками.

JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
var isEnabled = true;
var maxTabs = 1;
 
 
function updateBadgeText() {
    var tabsBalance = maxTabs - tabsCount;
    var tabsAllowanceRemaining = (tabsBalance > 0) ? tabsBalance : 0;
 
    chrome.browserAction.setBadgeText({
        text: "" + tabsAllowanceRemaining
        
    });
    
}
 
 
function getTabsCount(winId)
{
    return new Promise(function (resolve)
    {
        chrome.windows.getCurrent(function (win)
        {
            chrome.tabs.query({ windowId: winId ? winId : win.id }, function (tabs)
            {
                resolve(tabs.length);
            });
        });
    });
}
 
 
 
async function handleTabCreated(tab)
{
    let tabsCount = await getTabsCount(tab.windowId);
    if (tabsCount >= maxTabs)
    {
        setTimeout(function ()
        {
            chrome.tabs.remove(tab.id);
        }, 2000);
    }
}
 
 
 
 
function handleTabRemoved(tab,response)
{   
    getTabsCount(tab.windowId);
    if (response.isWindowClosing == false)
    {
            chrome.tabs.query({ "active": true, "windowId": response.windowId }, function (tabs)
            {
                chrome.tabs.reload(tabs[0].id, function ()
                {
                    began(tabs[0].id);
                
                });
            });
    }
}
    
 function began(tabId) {
    setTimeout( function() {
        chrome.tabs.sendMessage(tabId, {greeting: "hello"}, function(response) 
   {});
   }, 2000);
}
 
function handleTabUpdated(tab) {
    getTabsCount(tab.windowId);
}
 
 
function init() {
    getTabsCount();
    chrome.tabs.onCreated.addListener(handleTabCreated);
    chrome.tabs.onRemoved.addListener(handleTabRemoved);
    chrome.tabs.onUpdated.addListener(handleTabUpdated);
    }
 
function teardown() {
    chrome.tabs.onCreated.removeListener(handleTabCreated);
    chrome.tabs.onRemoved.removeListener(handleTabRemoved);
    chrome.tabs.onUpdated.removeListener(handleTabUpdated);
    }
 
chrome.browserAction.onClicked.addListener(function (tab) {
    if (!isEnabled) {
        init();
        chrome.browserAction.setIcon({ path: "icons/19.png" });
    }
    else {
        teardown();
        chrome.browserAction.setIcon({ path: "icons/19-disabled.png" });
        chrome.browserAction.setBadgeText({ 'text': '' });
  }
 
    isEnabled = !isEnabled;
});
 
init();
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 18:16
Строка 6 - ты все еще продолжаешь использовать удаленную переменную.
Строки 50, 72, 77 - ты просто вызываешь функцию, которая ничего не обновляет, а просто возвращает значение. Непонятно для чего ты это делаешь.
Функция должна вызываться только тогда, когда надо получить количество вкладок. Делается в теле асинхронной функции (определенной с помощью async) с использованием слова await. То есть, возвращаясь к 6-ой строке надо функцию переписать так
JavaScript
1
2
3
4
5
6
7
8
9
10
async function updateBadgeText() {
    var tabsBalance = maxTabs - await getTabsCount;
    var tabsAllowanceRemaining = (tabsBalance > 0) ? tabsBalance : 0;
 
    chrome.browserAction.setBadgeText({
        text: "" + tabsAllowanceRemaining
        
    });
    
}
Для чего в остальных местах функция вызывается я не понял, видимо там ее вызов надо просто убрать.

Относительно отладчика тут тоже есть неприятные новости: в консоль выводится только информация со страницы, из бэкграунда можно послать туда сообщение или вывести что-то с помощью tabs.ExecuteScript.
0
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 18:22  [ТС]
background.js:6 Uncaught SyntaxError: await is only valid in async function
Ругается на 6ую строку
var tabsBalance = maxTabs - await getTabsCount;
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 18:26
BeleK0ss, у меня функция асинхронная. Ты написал async в объявлении?
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 18:30  [ТС]
Забыл. Дописал, теперь не ругается.

Убрал, как я понял, лишнее. Теперь все выглядит вот так.
Кликните здесь для просмотра всего текста
JavaScript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
var isEnabled = true;
var maxTabs = 1;
 
 
async function updateBadgeText() {
    var tabsBalance = maxTabs - await getTabsCount;
    var tabsAllowanceRemaining = (tabsBalance > 0) ? tabsBalance : 0;
 
    chrome.browserAction.setBadgeText({
        text: "" + tabsAllowanceRemaining
        
    });
    
}
 
 
function getTabsCount(winId)
{
    return new Promise(function (resolve)
    {
        chrome.windows.getCurrent(function (win)
        {
            chrome.tabs.query({ windowId: winId ? winId : win.id }, function (tabs)
            {
                resolve(tabs.length);
            });
        });
    });
}
 
 
 
async function handleTabCreated(tab)
{
    let tabsCount = await getTabsCount(tab.windowId);
    if (tabsCount >= maxTabs)
    {
        setTimeout(function ()
        {
            chrome.tabs.remove(tab.id);
        }, 2000);
    }
}
 
 
 
 
function handleTabRemoved(tab,response)
{   
  
    if (response.isWindowClosing == false)
    {
            chrome.tabs.query({ "active": true, "windowId": response.windowId }, function (tabs)
            {
                chrome.tabs.reload(tabs[0].id, function ()
                {
                    began(tabs[0].id);
                
                });
            });
    }
}
    
 function began(tabId) {
    setTimeout( function() {
        chrome.tabs.sendMessage(tabId, {greeting: "hello"}, function(response) 
   {});
   }, 2000);
}
 
 
 
 
function init() {
    chrome.tabs.onCreated.addListener(handleTabCreated);
    chrome.tabs.onRemoved.addListener(handleTabRemoved);
    }
 
function teardown() {
    chrome.tabs.onCreated.removeListener(handleTabCreated);
    chrome.tabs.onRemoved.removeListener(handleTabRemoved);
    }
 
chrome.browserAction.onClicked.addListener(function (tab) {
    if (!isEnabled) {
        init();
        chrome.browserAction.setIcon({ path: "icons/19.png" });
    }
    else {
        teardown();
        chrome.browserAction.setIcon({ path: "icons/19-disabled.png" });
        chrome.browserAction.setBadgeText({ 'text': '' });
  }
 
    isEnabled = !isEnabled;
});
 
init();
0
 Аватар для diadiavova
7258 / 2605 / 744
Регистрация: 11.04.2015
Сообщений: 4,148
Записей в блоге: 43
16.01.2018, 18:37
BeleK0ss, У тебя макстаб равно единице, то есть если вкладка одна, окно всё равно закроется.
1
0 / 0 / 0
Регистрация: 27.12.2017
Сообщений: 23
16.01.2018, 18:49  [ТС]
Вроде заработало. Спасибо! Очень сильно выручил!
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
16.01.2018, 18:49
Помогаю со студенческими работами здесь

Как по ссылке перейти на нужную вкладку (tabs)?
Здравствуйте! На одной странице есть ссылки на проекты. На второй странице есть описание проектов в виде закладок (tabs). Как сделать...

Объяснить работу алгоритма std::remove();
Есть буквы: #include <iostream> #include <algorithm> #include <vector> int main () { std::vector<int> vec = {10,...

Реализовать аппликативный оператор MY-REMOVE-IF с интерфейсом и семантикой, аналогично стандартному REMOVE-IF
Реализовать аппликативный оператор MY-REMOVE-IF с интерфейсом и семантикой, аналогично стандартному REMOVE-IF.

Функция remove() удаляет только заранее запланированые файлы, выдавая ошибку на remove (STRING)
Салем, начал изучать файловую работу в С++, и столкнулся с такой проблемой, что функция remove() соглашается удалять только заранее...

Как исправить всплывающую рекламу в Google Chrome
Как исправить всплывающую рекламу в Google Chrome


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
19
Ответ Создать тему
Новые блоги и статьи
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Как объединить две одинаковые БД Access с разными данными
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
Создание Single Page Application на фреймах
krapotkin 16.11.2025
Статья исключительно для начинающих. Подходы оригинальностью не блещут. В век Веб все очень привыкли к дизайну Single-Page-Application . Быстренько разберем подход "на фреймах". Мы делаем одну. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru