— Хотел бы я иметь приложение, которое умеет что-то такое, что мне надо. Жаль такого нет.
— А почему бы тебе его самому не написать?
— Это сложно, мне потребуется куча времени, чтобы понять теорию, приступить к практике и, скорее всего, ничего хорошего не выйдет. А вообще я frontend-разработчик и привык к HTML, CSS и JS. Этот стек не позволяет писать десктопные приложения.
— Electron!
Уверен, большинство из нас обмениваются скриншотами. Существует большое количество приложений, способных делать скриншоты и как-то их редактировать (обрезать, рисовать что-то итд). Первые 3, которые мне приходят на ум: Joxy, Monosnap, Gyazo.
Казалось бы, чего ещё желать, когда есть задача и инструменты для неё? Но нет, всё не так просто.
Критерии, по которым я искал приложение
Сохранение скриншота на сервере
Намного удобнее скинуть ссылку на скриншот, чем искать, куда приложение сохранило картинку, а потом прикреплять её. Конечно, некоторым не понравится идея хранить скриншоты неизвестно где. Но, с другой стороны, это позволяет быстро решить задачу — поделиться скриншотом. Такая модель поведения приложения мне вполне подходит.
Рисование на скриншоте
По своему опыту могу сказать, что стрелка у меня присутствует чуть ли не на каждом скриншоте. Намного проще указать на какую-то деталь, чем писать что-то вроде: «Вот эта вот штука между...». Также в стандартный набор инструментов для рисования на скриншоте, по моему мнению, должны входить карандаш, прямоугольник и текст. При этом должна присутствовать возможность переместить, перевернуть и удалить нарисованный элемент.
Версия для Linux
Как оказалось, не так и много приложений существует для этой задачи под Linux. Долгое время я использовал Gyazo и использовал бы дальше, если бы за «рисовашки» на скриншоте не приходилось платить.
Из коробки
Приложение должно сразу работать так как мне надо. Я не хочу что-либо настраивать, регистрироваться на всяких Imgur и т. д. Просто хочу захватить скриншот, кропнуть, сохранить и поделиться ссылкой на него.
На протяжении года я периодически занимался поиском, но в итоге так и не нашёл приложение, удовлетворяющее моим требованиям.
Electron
Время шло и я решил поближе познакомиться с Electron, чтобы понять что он умеет в принципе. Для тех кто не в курсе, Electron — технология, позволяющая создавать десктопные приложения, используя HTML, CSS и JavaScript.
И вот одной из демок демонстрировалось получение скриншота экрана пользователя. Это означало следующее:
- можно наконец-то написать приложение, отвечающее моим требованиям,
- появилась возможность познакомиться с новой для себя технологией на реальном проекте
Куда идти, с чего начать, как оно работает
Эти вопросы я задаю себе каждый раз, когда хочу изучить что-то новое. Зачастую решение одно и то же — документация и какой-нибудь официальный «Get started». У Electron и с тем, и с другим всё в полном порядке. По крайней мере они так пишут на сайте проекта — «It's easier than you think». Это действительно так! Единственное, что надо сразу понять — приложение составляют 2 процесса: Main и Renderer. В Main мы работаем с окнами приложения, меню и т.д., а в Renderer у нас по-сути просто наша страничка.
С чего начать приложение
Сперва давайте создадим проект со следующей структурой:
— package,json
— main.js
— scripts
— renderer.js
Для начала нам этого вполне достаточно. В main.js у нас будет содержаться код, относящийся к главному процессу, а в scripts -> renderer.js рендер. Теперь необходимо создать само приложение. Для этого в main.js пишем:
const electron = require('electron');
const app = electron.app;
app.on('ready', () => {
createWindow();
});
app.on('window-all-closed', function() {
app.quit();
});
Что в итоге мы сделали? Мы подключили Electron, создали само приложение и описали два обработчика. Когда наше приложение будет готово, мы вызовем функцию создания нового окна. А в тот момент, когда все окна будут закрыты — убьём процесс приложения.
Окно
Окей, теперь у нас есть сам процесс приложения. Теперь хотелось бы создать для него окно. Помните мы вызывали функцию createWindow();? Так вот пришла пора описать её.
function createWindow() {
appWindow = new BrowserWindow({
width: 300,
height: 300
});
};
Также необходимо подключить:
const BrowserWindow = electron.BrowserWindow;
Таким образом мы создали окно с заданными размерами. Очень просто, не правда ли?
У окна есть очень много свойств, методов и событий, связанных с ним. К примеру: title, icon, x, y, show(), hide(), close(), minimize, closed, restore, resize. Перечислять весь список бесполезно, поэтому вот ссылка на Browser WIndow.
Но смотреть на пустое окошко не очень весело, да? Давайте подгрузим туда какую-нибудь простую страничку, которая содержит заголовок «Hello, world!». На уровне с main.js
создаём файл index.html, который выглядит как самая обычная страница с <h1>Hello, World!</h1>
. Теперь только остаётся расширить функцию createWindow(), дописав
appWindow.loadURL(`file://${__dirname}/index.html`);
Запускаем наше приложение и видим:
Прям какая-то магия :) Но на секундочку о грустном.
Дебаг
А как же собственно отлавливать ошибки? Всё максимально просто! Если ошибка в главном процессе, то она и вывалится вам в консоль, из которой вы запускаете приложение. Для ловли ошибок на самой странице мы будем использовать браузерную консоль. Да, именно так! Наше приложение по факту является страничкой, запущенной в браузере. Чтобы включить консоль давайте допишем в createWindow();
строчку: appWindow.webContents.openDevTools();
. Теперь при запуске приложения мы сразу же увидим консоль.
Меню
Пока что всё что мы получили — это страничка в окне. Хочется как-то взаимодействовать с ней, к примеру, с помощью меню. Для этого в Electron есть специальный класс Menu. Поставим себе задачу — создать меню из одного элемента, при нажатии на который мы увидим системное всплывающее окно с тайтлом «Hello», сообщением: «Do you like this?» и вариантами ответа ['Yes', 'No'].
В main.js пишем:
const appMenu = require('./scripts/appMenu');
И создаём файл appMenu.js в директории scripts. Внутри себя он содержит следующий код:
const { shell } = require('electron');
const dialog = require('electron').dialog;
const newShotDialog = {
type: 'info',
title: 'Hello',
message: 'Do you like this?',
buttons: ['Yes', 'No']
};
module.exports = function appMenu(app, appWindow) {
return (
[
{
label: 'File',
submenu: [
{
label: 'Click me',
click() {
dialog.showMessageBox(newShotDialog, function(index) {
})
},
},
],
},
]
);
};
Идём далее. В app.on('ready'...
дописываем:
const template = appMenu(app, appWindow);
const menu = electron.Menu.buildFromTemplate(template);
electron.Menu.setApplicationMenu(menu);
После запуска приложения мы увидим, что наше меню стало меньше и теперь содержит один элемент. Нажав на него, мы увидим следующее:
Собственно, мы решили поставленную задачу. Я мог бы посвятить всю статью созданию какого-либо абстрактного приложения (например, аудио-плеера), но тогда бы она вся состояла из «какой-то код» -> «результат», что, согласитесь, не очень интересно (лично я бы закрыл вкладку и пошёл писать что-то своё). Поэтому остальную часть статьи я уделю обзору некоторых важных, на мой взгляд, возможностей Electron на примере моего проекта --shots. Не беспокойтесь, приложение, создание которого я описал выше, можно найти по ссылке
Больше возможностей Electron на примере --shots
Согласитесь, не очень удобно каждый раз заходить в меню за инструментами которые вы часто используете (кроп, стрелка и т.д.). Решить данную проблему можно созданием панельки в области страницы, на которую их и вынести. Уже удобнее (не надо пробираться через громоздкое меню), но всё ещё не идеально (тянуться-то надо).
Можно лучше?
Конечно, можно! Давайте использовать контекстное меню. Его будет вызывать клик правой кнопкой мыши в области окна. Такое меню будет состоять из пунктов, каждый из которых при нажатии вызовет свою функцию. К примеру, я разместил в этом меню следующие элементы:
- New shot
- Save
- Default
- Crop
- Arrow
- Rect
- Pen
- Blur
Этот набор инструментов даст воспользоваться практически всеми возможностями приложения.
А может ещё лучше?
Да, можно ещё больше упростить доступ к возможностям приложения. Добьёмся мы этого при помощи хоткеев. Хоткеи регистрируются при помощи модуля globalShortcut и метода register. Необходимо помнить о различиях клавиатур в различных ОС. Так, например, слушать Command на Linux и Windows бесполезно. Поэтому разработчики Electron предлагают слушать CommandOrControl.
Хоткей в стандартном случае состоит из модификатора (CommandOrControl, Alt, Option...) и кода клавиши (0 — 9, A — Z, Space, Tab...). Стоит отметить, что он не обязательно должен состоять из комбинации двух клавиш. Может встретиться как одна, так и три (возможно и больше, но я не проверял). Например, для вызова модального окна со списком хоткеев я использую F2, а для того, чтобы увеличить масштабирование (zoom in) — CommandOrControl+shift+Plus
. Мне пришлось добавить в эту цепочку shift, потому что иначе комбинация CommandOrControl+shift+Plus
преобразуется в сontrol+shift+=
.
Общение процессов
Скорее всего, вы столкнётесь с необходимостью передать сообщение из одного процесса в другой (вызови мне такую-то функцию, к примеру). Для этого существуют ipcRenderer и ipcMain. Поставим себе простую задачу, чтобы разобраться как это работает. Допустим, при нажатии на пункт в меню мы хотим вызвать некую функцию, которая выведет нам alert. В обработчике клика пропишем что-то вроде appWindow.webContents.send('show');
. Это означает следующее: при нажатии на пункт меню в renderer-процесс будет послано асинхронное сообщение по каналу 'show'. Также можно передать дополнительные аргументы, так что пусть наш alert выведет и переданный аргумент. Немного модифицируем написанное ранее appWindow.webContents.send('show', 'content');
.
Теперь в renderer-процессе напишем:
ipcRenderer.on('show', (event, message) => {
alert(message) // Alerts 'content'
});
Как видите, всё очень просто!
Tray
Полезно чтобы приложение не висело свёрнутым или, что хуже, было просто открыто. Для решения этой проблемы существует трей. И Electron даёт возможность поработать и с ним через класс Tray.
Что по сути такое приложение в трее? Это иконка и меню, вызываемое кликом правой клавиши мыши. Интересно подметить, что конструктор класса Tray принимает ровно один параметр — иконку. Если этот параметр отсутствует, то будет выброшена ошибка. Сначала это ничуть меня не удивило. Есть в конструкторе обязательный параметр, будь добр — передай. Чуть позже расскажу почему в дальнейшем обязательность этого параметра мне показалась странной.
Иконка приложения
Давайте ещё немного поговорим об украшательствах. Как же наше приложение будет жить без иконки?! Добавляется она там же, где мы создавали новое окно, дописыванием в конец icon: __dirname + '/icon.png'. Стоит отметить, что для Windows рекомендуется использовать иконку в формате *.ico. Не страшно, ведь можно воспользоваться os.platform()
.
Моя страничка может достучаться до системы?
Да, конечно. На момент написания этой статьи актуальная версия Electron — 1.4.6, которая работает с NodeJS 6.5. К примеру, у --shots есть возможность локального сохранения файла. Для этого я использую fs.writeFile
. И каждый раз перед таким сохранением проверяю есть ли директория для скриншотов (я назвал её «--shots»). Если она отсутствует, то просто её создаю.
Очень важно понимать, что ваше приложение на Electron — это не просто страничка в стандартном окошке, а вполне себе полноценное приложение.
Дополнительные возможности, которых нет в --shots
Естественно, работая над --shots, я использовал не все возможности Electron.
Например, можно загрузить своё приложение в AppStore и WindowsStore. Но --shots задумывался как приложение для LInux, поэтому особо я не влезал во все эти дебри публикации приложения. Но знаю, что для того чтобы выложить приложение в AppStore, необходимо его подписать. Каждый раз при сборке новой версии --shots я вижу этот warning в консоли и скрещиваю пальцы, чтобы он ни на что не повлиял в дальнейшем.
Ещё на сайте Electron можно встретить один замечательный пункт: «Automatic updates». Но и его я не смог прикрутить к --shots, т.к. открыл статью и увидел:
«There is no built-in support for auto-updater on Linux, so it is recommended to use the distribution’s package manager to update your app».
Проще говоря, на LInux это не работает, так что выкручивайтесь. Например, я делаю запрос на сервер, который возвращает мне номер текущей актуальной версии и, если он не совпадает с версией приложения, вывожу сообщение с просьбой его обновить.
Подводные камни
Вроде, звучит всё круто, должен же быть подвох. Безусловно, есть некоторые неприятные моменты:
Не всё кроссплатформенно
Это заметно даже на примере того же самого авто-апдейта. С прозрачным окном тоже не всё так просто. К примеру, на Linux нужно вбить в командную строку --enable-transparent-visuals --disable-gpu
, а иначе никакого прозрачного окна вы не увидите. Согласитесь, приложение, которое после установки просит пользователя вбить что-то в терминал, уже начинает вызывать подозрение.
Minimize
Очень много времени я потратил на разрешение проблемы с этим. Суть её такова: когда мы сворачиваем окно приложения — срабатывает событие minimize, и окно сворачивается. Вроде всё окей. Но приложение, висящее и в трее, и в доке — это уже странно. Поэтому хотелось бы отлавливать minimize и как-то убирать приложение из дока, оставляя лишь в трее. Для этого есть метод hide(). Пишем обработчик для minimize, вызываем hide — всё отлично. Затем я захотел добавить возможность из меню трея развернуть приложение обратно, и сразу же нашёл метод show(). Всё логично show/hide, но нет. Когда я пытаюсь развернуть приложение из трея, оно намертво зависает. Очень долго я думал, что упускаю что-то важное, но никаких ошибок в консоли не видел. В том числе, выводил само окно перед тем как вызывать show() — окно существует. В общем, отказался от подобного механизма работы приложения и решил попробовать вызвать hide(), когда окно приложения открыто. И да, чудо свершилось. Всё заработало ровно так, как надо. Несколько часов было потрачено на поиск ответа на вопрос: «Да почему ты не работаешь?», а в итоге решение мне подсказал Telegram. Я просто добавил в меню пункт «minimize to tray», а заодно и хоткей повесил на него.
В общем-то для меня подводные камни Electron’а закончились. Скорее всего, мне просто повезло…
Сборка
Настало время поговорить об ещё одной достаточно занимательной вещи — сборке. Electron даёт вам возможность собрать своё приложение и потом проинсталлировать его в различных ОС. Пользователь даже может не догадываться что вы написали своё приложение на web-технологиях. Давайте приступим!
Стандартный подход
На официальном сайте в разделе документации есть 3 ссылки на инструкции по сборке (Linux, MacOS, WIndows). Так как приложение изначально затачивалось под Linux, то сначала я открыл ссылку для него. Первое системное требование сразу же напугало меня: «At least 25GB disk space and 8GB RAM» (не менее 25GB свободного места и 8GB оперативки). «Ладно, что поделать?» — подумал я и начал пытаться собрать приложение из примера. В итоге, сам процесс сборки занял у меня минут 20-30, более того, пользоваться компьютером было невозможно! Я очень рад, что не продолжил работать с этим сборщиком, т.к. узнал об одной пренеприятнейшей вещи: чтобы собрать приложение под какую-то ось, его надо собирать именно из под этой оси!
«Должен быть другой способ,» — говорил я себе. И да, действительно он существует.
Electron builder
Достаточно было просто загуглить «electron builder» и перейти по первой ссылке. С тех самых пор я и использую это сборщик, и вот почему:
- сборка --shots для Linux стала занимать ~ минуту,
- из-под Linux можно собрать и .exe-шник (надо ещё Wine ставить, но ок),
- в процессе сборки можно смотреть мемы, ничего не тормозит.
Всё что вам остаётся — правильно дополнить свой проект.
Подготовка проекта
Этот процесс можно разбить на 2 этапа:
- подготовка иконок,
- подготовка package.json.
Иконки
Приложения, будучи установленными, хотят иметь свои иконки. Для этого в корневой директории проекта необходимо создать директорию «build», которую заполнить по следующей схеме:
— build
—— icons
——— 32x32.png
——— 32x32.png.ico
—— icon.icns
—— icon.ico
Package.json
Если вы просто попробуете собрать свой проект, то консоль начнёт плеваться в вас ошибками. Это происходит по той причине, что в файле должны содержаться стандартные строки:
- name,
- description,
- version,
- author.
Затем надо описать «build» и «scripts». Я намеренно быстро пробежался по этим двум пунктам, т.к. запоминать, что внутри, не требуется, да и вряд ли возможно, особенно если учесть, что есть огромное количество настроек (для «build»), которые могут зависеть от ОС, к которой он и собирается. Например, для Windows можно собрать portable-версию своего приложения и указать gif-файл, который будет отображаться в момент запуска.
В общем, рекомендую использовать именно Electron builder для сборки. Это сэкономит вам кучу нервов, поверьте.
Добавляем своё приложение на сайт Electron
Я считаю неплохой идеей рассказать о своём проекте на Electron на сайте самого Electron. Для этого достаточно послать им pull request, содержащий:
- отредактированный «_data/apps.yml» (добавив ваше приложение в конец);
- иконку приложения 256px на 256px в формате png;
- сообщение о том, что вы связаны с проектом и все члены вашей команды согласны с тем, что приложение появится на сайте Electron.
Опять же, в этом нет ничего сложного. После отправки pr необходимо лишь запастись терпением и ждать когда его примут (в моём случае это заняло около недели).
Подробную инструкцию по добавлению можно найти здесь.
Подводя итог
Нет, Electron не вытеснит нативщиков с рынка, так же как и React Native не убьёт Swift. Electron годится для создания простых приложений и является достаточно интересным проектом. И он чертовски прост! Мне удалось написать --shots, используя стандартный стек технологий frontend-разработчика:
- HTML
- CSS
- PostCSS
- JavaScript
- NodeJS
- PHP
Первый коммит был сделан 24 сентября 2016 года, а первый релиз вышел 17 октября того же года. Это значит, что спустя всего 24 дня я получил версию приложения, которую уже можно было использовать. Это достаточно быстро, если учесть, что я работаю и люблю отдыхать. Так что вперёд, всё в ваших руках!
Полезные ссылки
Комментарии (46)
stepanp
05.12.2016 18:10-10Это сложно, мне потребуется куча времени, чтобы понять теорию, приступить к практике и, скорее всего, ничего хорошего не выйдет. А вообще я frontend-разработчик и привык к HTML, CSS и JS.
Людям, для которых изучение нового стека является проблемой, не место в индустрии.
И уж тем более не повод тащить любимый набор говна и палок в натив.
bingo347
05.12.2016 19:00Читая статью вспоминал, как ковырял свое первое приложение на электроне чуть больше года назад (документация тогда была на порядок запутаннее, чем сейчас). Вроде бы и выглядит как пересказ основных моментов из документации, но сделано это качественно и доступным языком, за что хочу Вас поблагодарить.
Думаю было бы полезно в подобном же стиле расписать работу со следующими стандартными библиотеками электрона:
web-view-tag
clipboard
Особенно интересует webview так как позволяет запустить отдельный render процесс в том же рабочем окнеBinjo
05.12.2016 19:18Спасибо! С clipboard у меня как раз и идёт работа, когда при сохранении скриншота возвращается ссылка. Заключается она ровно в двух строчках:
const {clipboard} = require('electron') clipboard.writeText('Example String')
Поэтому я и опустил этот момент :)
bingo347
05.12.2016 20:05Насчет clipboard написал, так как в свое время долго мучился, как организовать копирование файлов в файловом менеджере (например nautilus в ubuntu) и вставку их в моем приложении, при абсолютной уверенности, что это возможно, так как Atom и VSCode написанные на электроне это умеют, пока методом тыка не ввел в консольке clipboard.readText() и получил строку с абсолютными путями до файлов
rvt
05.12.2016 19:09http://makescreen.ru
Отсутствие десктопного приложения, никакой рекламы и полный минимализм.Binjo
05.12.2016 19:15+1Это по сути сайт с
input[type="file".
Открыть сайт, сделать самому скрин и закинуть скрин — слишком много действий для того, чтобы просто перейти к редактированию. И неизвестно как он себя поведёт в оффлайне, т.к. мне этот сайт каждый раз говорит, что я вставляю не изображение. Хотя действую по его инструкции.
terrakok
05.12.2016 19:29Напоминает историю с вк аудио плеерами. Каждый пишет свой. Хотя, как обучалка для начинающих знакомство с Electron сойдет.
23rd
05.12.2016 20:02+2— Хотел бы я иметь приложение, которое умеет что-то такое, что мне надо. Жаль такого нет.
— А почему бы тебе его самому не написать?
— Это сложно, мне потребуется куча времени, чтобы понять теорию, приступить к практике и, скорее всего, ничего хорошего не выйдет. А вообще я frontend-разработчик и привык к HTML, CSS и JS. Этот стек не позволяет писать десктопные приложения.
— Electron!
Очень классно, да. Особенно когда все подряд теперь на электроне делают свои «десктопные» приложения.
VK messenger, Discordapp, новый Skype для Linux, GIU Livestreamer, Visual Code, WhatsApp и многое-многое другое. Вот так пользуешься всем этим, а потом понимаешь, что у тебя сейчас запущенно пять одностраничных хромиум браузеров.
До коле? Будущее уже наступило?Binjo
05.12.2016 20:46И об Atom не стоит забывать. Тут смотря под какие цели оно надо. Писать убийцу Sketch, к примеру, точно не стоит вместе с Electron. Что-то более-менее простое — почему бы и нет. И опять же вопрос выгоды для бизнеса. Есть у меня предположение, что простенькие вещи таким способом написать дешевле.
pmcode
05.12.2016 20:47А с БД работать умеет? А через ORM? Как-то очень мало инфы на эту тему у гугла оказалось.
Binjo
05.12.2016 20:52Так глубоко не смотрел. У меня сохранение скриншота — просто AJAX-запрос, далее PHP и, при необходимости, сэйв в базу в виде
fulename: user
(очень условно так).
Но если учесть, что приложение по сути — chrome спереди, а nodejs сзади, то, наверное, да.
Опять же повторюсь, вопрос этот особо не рассматривал.
bingo347
05.12.2016 22:20+1все что можно сделать в node.js — можно и в electron, в том числе и работать с БД
megahertz
06.12.2016 06:40Для автообновления под Linux можете попробовать https://github.com/megahertz/electron-simple-updater
dolphin4ik
06.12.2016 09:25А после сборки всё так же много весит? 50? 60? Мб
Binjo
06.12.2016 10:48shots-1.0.0-amd64.deb
занимает 32.9 метра. Достаточно много, но не 50-60.shots.Setup.1.0.0.exe
— 32.5 метра.
В настройках сборщика можно указать степень сжатия. Возможно, 50-60 появляются там, где установлено не правильно.
VovanZ
06.12.2016 15:15На тему мотивации: всё, кроме сохранения скриншота на сервере умеет Shutter. На линуксе ставится из репозитория двумя командами (сначала надо установить сам Shutter, потом поставить необязательные зависимости, чтобы заработало редактирование).
искать, куда приложение сохранило картинку, а потом прикреплять её
В случае с тем же Shutter, картинку можно скопировать в буфер обмена и вставить сразу куда надо, многие сервисы это поддерживают (точно поддерживают: slack, telegram, vk).
На тему автообновления: мне не нравится идея, что какое-то левое приложение будет что-то там само обновлять.
Правильный способ для линукса: поднять свой репозиторий и обновлять средствами системы, через пакетный менеджер.Binjo
06.12.2016 15:19+1всё, кроме сохранения скриншота на сервере
В том-то и суть.
картинку можно скопировать в буфер обмена
--shots даёт такую же возможность. Копируй как хочешь, куда хочешь или локально сохраняй.
мне не нравится идея, что какое-то левое приложение будет что-то там само обновлять
Ну само-то оно не будет. В моём случае пользователь должен пойти и скачать самостоятельно. А так — Slack приходит вместе в паком обновлений. Захотел — не обновил. Всё на усмотрение пользователя.
А вообще статья не про --shots :)
23rd
06.12.2016 15:29В том-то и суть.
Ну так написали бы отправку на сервер как консольное приложение и вызывали в шаттере после того, как изображение скопировалось.
А вообще статья не про --shots :)
А про то, что скриншотер в 30 Мб это нынче нормально.Binjo
06.12.2016 15:42+1Ну так написали бы отправку на сервер как консольное приложение
И при переходе на другую ось мне пришлось бы снова писать его.
А про то, что скриншотер в 30 Мб это нынче нормально.
Если учесть что сам его код занимает 2 Мб, а остальное прилетает после сборки(из за Хрома на борту), то да. Более того, мы живём в 2016 и 30 метров это явно не проблема.
SPAHI4
06.12.2016 16:08Консольное приложение как правило это небольшой скрипт. Под mac/linux будет почти одно и то же. А под винду переписать уж всяком проще, чем делать целое приложение.
23rd
06.12.2016 21:39И при переходе на другую ось мне пришлось бы снова писать его.
При переходе на винду ставим ShareX и забываем абсолютно про все проблемы.
23rd
06.12.2016 15:30Увы Shutter нереально лагуч для скриншотера и иногда слишком много себе позволяет выжирать ресурсов.
seka19
07.12.2016 09:46На протяжении года я периодически занимался поиском, но в итоге так и не нашёл приложение, удовлетворяющее моим требованиям.
А что насчёт онлайн сервисов? Есть, например, http://pastenow.ru/ и ещё несколько подобных.
Binjo
07.12.2016 09:53Как-то он, прямо скажем, не очень. Опять же, чтобы сделать скриншот и получить возможность редактирования необходимо заскринить всю область, потом загрузить его, затем нажать на кнопку редактирования и только потом выбрать нужный инструмент. Слишком много действий. Да и не хочу вкладкой или окном держать этот сервис. Уж лучше у меня в трее приложение будет висеть. И работает тормознуто :(
Spiritschaser
07.12.2016 21:50Раньше для этого использовали дельфи. Или Tcl/Tk.
sigmanor
11.12.2016 18:42+1Для написания кроссплатформенных приложений?
Spiritschaser
13.12.2016 09:23Что не так с Tcl/Tk в плане кросс-платформенности и приложений?
Ну дельфи кроссплатформенный лазарус заменил.
prostofilya
Вообще-то, вами упомянутый joxy и бесплатен и под линукс есть.
Binjo
Да, бесплатен. Но если взглянуть в бесплатный тариф, то становится всё ясно. Наличие рекламы меня крайне не устраивает.
prostofilya
Но она же не встраивается в скриншоты. Не буду спорить, мне просто хранение скриншотов не пригождалось, а других причин я не вижу.
Binjo
Да, не спорю. Просто реклама акцентирует внимание. Да, люди зарабатывают этим деньги и это их полное право. Мне же было интересно решить свою задачу и изучить что-то новое :)