Форум программистов, компьютерный форум, киберфорум
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. Показов 2249. Ответов 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
7261 / 2608 / 745
Регистрация: 11.04.2015
Сообщений: 4,155
Записей в блоге: 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
Ответ Создать тему
Новые блоги и статьи
Midnight Chicago Blues
kumehtar 24.03.2026
Такой Midnight Chicago Blues, знаешь?. . Когда вечерние улицы становятся ночными, а ты не можешь уснуть. Ты идёшь в любимый старый бар, и бармен наливает тебе виски. Ты смотришь на пролетающие. . .
Контроль уникальности заводского номера - вариант №2
Maks 24.03.2026
В отличие от предыдущего варианта добавлено прерывание циклов, также добавлены новые переменные для сохранения контекста ошибки перед прерыванием цикла: Процедура ПередЗаписью(Отказ, РежимЗаписи,. . .
SDL3 для Desktop (MinGW): Вывод текста со шрифтом TTF с помощью библиотеки SDL3_ttf на Си и C++
8Observer8 24.03.2026
Содержание блога Финальные проекты на Си и на C++: finish-text-sdl3-c. zip finish-text-sdl3-cpp. zip
Жизнь в неопределённости
kumehtar 23.03.2026
Жизнь — это постоянное существование в неопределённости. Например, даже если у тебя есть список дел, невозможно дойти до точки, где всё окончательно завершено и больше ничего не осталось. В принципе,. . .
Модель здравоСохранения: работники работают быстрее после её введения.
anaschu 23.03.2026
geJalZw1fLo Корпорация до введения программа здравоохранения имела много невыполненных работниками заданий, после введения программы количество заданий выросло. Но на выплатах по больничным это. . .
Контроль уникальности заводского номера - вариант №1
Maks 23.03.2026
Алгоритм контроля уникальности заводского (или серийного) номера на примере документа выдачи шин для спецтехники с табличной частью в КА2. Данные берутся из регистра сведений, по которому настроено. . .
Хочу заставить корпорации вкладываться в здоровье сотрудников: делаю мат модель здравосохранения
anaschu 22.03.2026
e7EYtONaj8Y Z4Tv2zpXVVo https:/ / github. com/ shumilovas/ med2. git
Программный отбор элементов справочника по группе
Maks 22.03.2026
Установка программного отбора элементов справочника "Номенклатура" из модуля формы документа в КА2. В качестве фильтра для отбора справочника служит группа номенклатуры. Отбор по наименованию. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru