1
  1. Все вопросы задаем в соответствующих темах Спасибо
  2. ВНИМАНИЕ сдесь живет Ссылка откроется в новом окне на конвертер плейлистов Милости просим

FAQ Пишем плагин сами

Тема в разделе "ПЛАГИНЫ", создана пользователем kil1970, 06-11-2017.

  1. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    сразу оговорюсь что тема не моя , ее любезно создал и дал добро на размещение автор, Ака Учкун
    дуумаю она будет полезна многим :)
    кому в плане прочтения (чтоб сразу забыть :)), кому для дальнейших наработок , ну и т.д.
    Пысы ну и чтоб не переделывать многое в сообщениях
    разговор будет как бы от первого лица :)
    ==========================================
    Пишем плагин сами или случайные уроки по python

    В чем смысл данной темы?
    Ведь не настолько же я наивен, чтобы полагать, что можно вот так взять и с помощью нескольких постов на форуме, научить простого пользователя ресиверов с имиджем на энигме писать плагины.
    Да, разумеется это невозможно.
    Цель этого FAQ вижу в другом.
    Исходя из своего опыта изучения языка программирования python, знаю, что самое неприятное в этом деле это при всей этой для начала очень нудной зубрежке, совершенное непонимание того, что как же эту чертову теорию соотнести применительно к нашим имиджам на энигме.
    То есть как недавно сказал один пользователь
    Что же понять его можно.

    Вот и постараюсь показать как соотнести, даже еще не самое возможно углубленное изучение языка python с практикой.
    И возможно затем, получив некое вдохновение вы возьметесь-таки за более углубленное изучение python.
    То есть вы поняли, задача этого FAQ, сделать процесс изучения python веселее, если кто решился-таки изучать его ради того, чтобы научиться писать или хотя бы править плагины...........

    Итак, начнем.
    Для начала определим свою задачу.
    Чтобы было наиболее наглядно, нам с вами нужно написать плагин (прямо в этой теме), в котором бы отображался весь необходимый процесс плагинописания. Сам питоновский код и все необходимые действия с имиджем.
    Да и еще, чтобы такого плагина еще не было бы написано (иначе какой смысл) и чтобы он был полезен для нас.

    И вот какой плагин мы напишем.
    Многие из нас частенько устанавливают имиджи с нуля.
    При этом мы довели этот процесс до автоматизма.
    Установили имидж, быстренько накатили настройки и пользуемся.
    Так вот, кому как, а мне лично каждый раз при этом приходится вручную переписывать файл settings в имидже, чтобы внести туда настройки тюнера. Много "тарелок", сложное подключение спутников к тюнеру и через дайсики протокола 2.0, а затем в конце еще и дайсик протокола 1.1.
    Дримбоксэдит передает все настройки кроме настроек тюнера.
    Конечно переписать вручную файл сеттингс вроде нетяжело и недолго, но почему бы не автоматизировать этот процесс? Тем более что есть пользователи, которые как огня боятся что-нибудь в имидже править вручную.

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

    Какой плагин написать решили, теперь нам надо подготовить софт, с помощью которого собственно и будем творить.

    Разумеется сперва-наперва устанавливаем у себя на компе собственно саму среду программирования python.
    Как и любое свободное ПО, python скачиваем с официального сайта, то есть вот отсюда . Пожалуйста войдите или зарегистируйтесь
    В наших имиджах (практически во всех) установлен Python версии 2.7.
    Потому именно эту версию питона нужно скачать и установить.
    Как скачать с сайта и как установить думаю показывать нет необходимости, ибо если не разберетесь с этим, далее эту тему читать нет смысла.....

    После установки python, если вы работаете на винде, обязательно проверьте, а добавился ли путь к python в переменную path.
    Для этого открываем
    Панель управления-Система
    Далее во вкладке Дополнительно или Дополнительные параметры системы (смотря какая у вас винда) находим кнопочку Переменные среды и нажав на нее, видим это окошко
    59a4466afb8e.png
    И в нижнем окне нажав на пункт "Path" нажимаем Изменить... и проверяем, есть ли там путь к python. Если нет, добавляем, есттественно узнав сначала, куда же установился python.
    У меня например он установился в папку C:\Python27 и я вот так добавил путь к нему в переменную path, просто прописав в начало этой переменной C:\Python27\; то есть отделил этот путь от остальных записей точкой с запятой.....
    5eb271166eaa.png

    Собственно в IDLE самого питона мы не будем писать плагин, так как есть более юзабельное ПО для этого.
    Но для компиляции на компе-то, да и для работы в консоли винды нам питон нужен будет........

    Сам плагин мы будем писать в очень продвинутом свободном (опенсурч) редакторе под названием Geany.
    Последнюю версию всегда можете скачать здесь Пожалуйста войдите или зарегистируйтесь
    Запускаем этот самый Geany и настраиваем так, чтобы эта продвинутая прога удобно подсвечивала нам синтаксис питона.
    Идем во вкладку Правка-Настройки-Редактор-Отображение и ставим настройки как показана на скрине. Здесь главное пункт - Обратить цвета подсветки синтаксиса (будет подсвечивать синтаксис питона очень удобно для чтения)
    32456a3d2861.png

    Пожалуй это вся подготовительная работа.
    Далее приступим к написанию непосредственно кода.

    Продолжение следует (по наличию свободного времени, сегодня или завтра)......
     
    Последнее редактирование: 18-01-2018
  2. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    Продолжим.
    Но сначала, как водится, снова лирическое отступление.
    Enigma2 на наших ресах это операционная система (ОС) на линуксе, такая же как Ubuntu, Mandriva, Slax и т.д.
    Хотите продвинуться дальше, изучайте устройство этой ОС енигма2, вникайте во все исходники.
    А уж где система (имидж) хранит свои настройки, а откуда из имиджа можно вытянуть какие сведения это нужно знать подавно.
    Вобщем потихоньку вместе с изучением питона, изучайте и имидж......

    Также надеюсь, хоть немножко "позубрили" питон, например хотя бы здесь . Пожалуйста войдите или зарегистируйтесь
    Хотя бы азы, что такое строки, списки, словари и их встроенные методы, простые операторы if-else, for, выражения, кортежи и т.д........
    В этой теме невозможно охватить все, будем выполнять только свою задачу, как использовать получаемые знания на имидже, и подразумевается, что питон вы уже изучаете.......

    Начнем решать нашу задачу.
    Вкратце повторим суть задачи.

    Задание.
    - написать плагин для наших имиджей на энигме2, который бы одним кликом сохранял настройки нашего тюнера, затем будучи установленным на другой "чистый" имидж, одним кликом восстановить настройки тюнера. Не ахти какой нужный плагин))), но суть - такого плагина нет и мы его напишем.

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

    1. Вытащить из имиджа настройки тюнера, и сохранить эти настройки в файл.
    2. Сохранить плагин с этими настройками и в последующем установить плагин с этими сохраненными настройками на вновь установленный нами имидж, ну и применить эти настройки в новом имидже.

    Вроде бы все просто. Вот теперь оба этих пункта по очереди детально будем осуществлять.
    Пункт первый, так как же вытащить из имиджа настройки тюнера?
    Вот мы и начали работать непосредственно с имиджем.
    В результате изучения энигмы2 узнаем, что настройки тюнера (тюнеров) имидж хранит (и использует оттуда) в файле
    [cut]/etc/enigma2/settings[/cut]

    Далее вступают в дело наши (пусть даже пока скудные) знания в питоне.
    Теперь разработаем непосредственно алгоритм по пункту номер один.

    а) открыть файл /etc/enigma2/settings для чтения
    б) прочитать файл построчно
    в) закрыть файл
    г) выделить строки с настройкой тюнера (тюнеров)
    д) открыть другой файл (вернее создать) с правами на запись и записать туда эти строки с настройками тюнера
    е) закрыть файл

    Как видите все очень просто и код будет простой.
    Открываем Geany, идем во вкладку Файл-Создать из шаблона, выбираем main.py.
    Откроется это окно.

    0e7a57b7307d.png
    Далее идем во вкладку Документ-Установить окончания строк, выбираем "Заменить окончания строк на LF (Unix)".
    Далее из окна удаляем готовый код и начнем.
    По порядку

    а) открыть файл /etc/enigma2/settings для чтения.

    Код:
        a = open("/etc/enigma2/settings", "r")
    Результат открывания файла присвоили переменной a (просто произвольная буква).

    Далее прочитываем файл
    б) прочитать файл построчно

    Код:
        b = a.read()
    Опять присваиваем переменной b (снова любой произвольный знак).
    Здесь нужны разъяснения.
    Как питон читает текстовый файл, состоящий из строк?
    Он читает его как одну единую строку, заменяя переходы на следующую строку на знак обратного слеша с латинской буквой n, то есть \n.
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Можем это легко проверить.
    Открываем файл /etc/enigma2/settings и смотрим. У меня он выглядит так:
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    А теперь вводя предыдущий код, читаем этот файл через python (заодно научимся работать непосредственно в питоне).
    Для этого открываем любую телнетовскую прогу.
    Например телнет в програмке DCC и логинимся в имидже:
    3d6d2bb06151.png

    И запускаем питон, предустановленный в имидже, просто написав команду python



    И вводим тот самый код по очереди
    a = open("/etc/enigma2/settings", "r")
    b = a.read()

    а затем попросим питон вывести на экран значение переменной b (то есть просто пишем b и нажимаем Enter)

    Ввели? Убедились?

    Далее обязательно закрываем файл, который открывали ранее для чтения (никогда не забывайте сделать это)
    в) закрыть файл
    Код:
        a.close()
    А вот следующий пункт выделить строки с настройкой тюнера (тюнеров) выполнить не так просто ввиду именно того, что результат чтения это только одна единая строка.
    Это один из нюансов работы со строками. Конечно существует много способов обойти эту проблему. Рассмотрим один из них.
    Для этого воспользуемся встроенным методом строк split, который создает список из строк, разделив именно на этих знаках \n (удалив эти знаки), снова присвоив результат какой-то переменной

    Код:
        c = b.split('\n')
    В результате получим список, состоящий из строк (из тех самых строк файла settings).
    И желательно этот код продолжаем вводить в телнете (где запустили питон в имидже).
    Это нужно еще, чтобы убедиться что какую-нибудь синтаксическую ошибку не допускаем.
    Я всегда так делаю, пишу код и тут же ввожу в питоне.....
    Можете также "приказать питону" вывести результат c и получите примерно такой результат

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Вот теперь можем из этого списка выдергивать члены-строки, которые относятся к настройкам тюнера (тюнеров).
    Для этого воспользуемся другим встроенным методом строк __contains__, который выделяет строки по содержанию, проверяя наличие текста в строке, который передан этому методу в качестве аргумента.
    А в файле сеттингс, как мы можем легко выяснить, абсолютно все строки с настройками тюнеров содержат "слово" Nims.
    То есть в этом случае мы включаем логику. И вообще, учтите в программировании без логики очень-очень трудно.
    С логикой надо очень сильно дружит.........
    Значит чтобы "выделить" настройки тюнеров, мы сначала создаем пустой список, а потом добавляем туда все строчки-члены предыдущего списка, которые содержат "слово" Nims. Применив оператор for для этого.

    г) выделить строки с настройкой тюнера (тюнеров)

    d = []
    for x in c:
    if x.__contains__('Nims'):
    d.append(x)

    Использовали еще один встроенный метод (но уже списка теперь) append.
    Все это простые операторы и простые методы строк и списков.
    Предыдущий код звучит так:
    "Каждый элемент х из списка с, если он содержит подстроку Nims, добавить в новый список d".
    В итоге мы получили список из строк с настройками тюнера (тюнеров).
    Но список мы не можем записать в файл как строки. Соответственно теперь проведем обратную операцию, воспользовавшись другим встроенным методом join. Этот метод прямая противоположность методу split, склеивает строки из списка, вот так:

    Код:
        e = '\n'.join(d)
    Так как питон требует в конце файла пустую строку, логика нам подсказывает, что к полученному нам бы нужно добавить символ переноса строки, то есть (включите логику и поймете, что я прав):

    Код:
        f =e + '\n'
    Давайте взглянем в окно Geany и проверим, одно и то же ли мы с вами пишем. Чем хорош Geany, вам стоит поставить двоеточие и нажать Enter, табуляция (или пробелы, смотря как настроите) любезно будет вставлена самой прогой.

    14d1c7f936ce.png

    Теперь весь этот код можете ввести в окно терминала, где вы вошли в питон на имидже (по одной строке), а вконце введите f и увидите, что результатом является одна строка в памяти....

    Далее создаем другой файл и записываем туда настройки тюнера.
    Так как пока просто пишем код, а не плагин непосредственно, пока будем создавать файл в папке темп (чтобы проверить код) (файл скажем назовем mysettings).
    То есть открываем (создаем) файл в папке темп с правами записи и записываем туда последнюю полученную строку:

    Код:
        g = open("/tmp/mysettings","w")
        g.write(f)
    И закрываем файл (обязательно)

    е) закрыть файл

    Код:
        g.close()
    Полученный код.

    2add6614be3e.png

    И введя весь этот код в терминале имиджа с вводом в питон, убеждаемся, что да код верный, вот он файл и его содержание.

    d824b56071e8.png

    1d3952e83002.png

    Конечно, полученный результат пока мало похоже на плагин, но дойдем и до этого......


    Продолжение следует......
     
    Последнее редактирование: 18-01-2018
  3. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    С кодом, с помощью которого будем сохранять настройки тюнера (тюнеров) разобрались.
    Теперь из этой второй части
    2. Сохранить плагин с этими настройками и в последующем установить плагин с этими сохраненными настройками на вновь установленный нами имидж, ну и применить эти настройки в новом имидже.
    будем разбираться с кодом, который будет восстанавливать настройки на вновь установленном имидже.


    То есть оформление непосредственно самого плагина оставим напоследок. Для любой программной задачи, главное код.

    С помощью значит этого кода, мы поняли, что сохраним настройки
    Код:
        a = open("/etc/enigma2/settings", "r")
        b = a.read()
        a.close()
        c = b.split('\n')
        d = []
        for x in c:
           if x.__contains__('Nims'):
              d.append(x)
        e = '\n'.join(d)
        f = e + '\n'
        g = open("/tmp/mysettings","w")
        g.write(f)
        g.close()
    Конечно при оформлении плагина изменим путь сохранения настроек с папки темп в другую нужную папку.
    Затем пользователь получается сохранить физически плагин с настройками куда нибудь на комп скажем, установит имидж, скопирует папку плагина скажем в папку /usr/lib/enigma2/python/Plugins/Extensions/, перегрузится, откроет окно плагина и выберет восстановление настроек.
    Вот сейчас нужно написать код, который и будет задействован после именно этого выбора восстановления пользователем.

    Алгоритм в данном случае также достаточно прост (вроде бы):
    а) открыть ранее сохраненный файл на чтение
    б) прочитать файл
    в) закрыть файл
    г) открыть файл /etc/enigma2/settings,
    д) удалить из него строчки, касающиеся настроек тюнера
    е) и наоборот записать туда ранее сохраненные строчки с настройками тюнера.

    По пунктам будем решать задачу. Пишем код.

    а) открыть ранее сохраненный файл на чтение
    Пока мы сохраняли файл в папку темп, следовательно

    Код:
        a = open("/tmp/mysettings", "r")
    Ну и прочитать файл
    б) прочитать файл

    Код:
        b = a.read()
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    в) закрыть файл

    Код:
        a.close()
    А вот с пунктами г, д, е немного незадача.
    То есть согласно выработанному нами алгоритму, мы должны открыть файл /etc/enigma2/settings и удалить из него определенные строчки, и наоборот записать туда другие строчки из нашего ранее сохраненного файла.
    А так как мы уже знаем, что файл settings это текстовый файл состоящий из последовательности символов (переход на другую строчку тоже символ \n), то бишь это строковый файл, ее нельзя изменять. Из питона знаем строка неизменяемая последовательность.
    Соответственно, исходя из этого обстоятельства мы перепишем наш последний алгоритм действий:

    а) открыть ранее сохраненный файл на чтение (уже выполнили)
    б) прочитать файл (уже выполнили)
    в) закрыть файл (уже выполнили)
    г) открыть файл /etc/enigma2/settings для чтения
    д) удалить из него строчки, касающиеся настроек тюнера
    е) и наоборот записать туда ранее сохраненные строчки с настройками тюнера.
    д) прочитать файл /etc/enigma2/settings
    e) закрыть файл /etc/enigma2/settings
    ж) методом split создать список строк
    и) удалить из этого списка те строки в которых есть подстрока Nims (списки как известно можно изменять)
    к) Список с оставшимися строками с помощью метода join вновь объединить в строку
    л) методом простой конкатенации соединить строки полученные из файлов /etc/enigma2/settings и /tmp/mysettings и записать новый файл settings
    м) этим новым файлом заменить файл /etc/enigma2/settings


    Ого, задача как видим усложнилась (из-за неизменяемости строк в питоне), но она все же по-прежнему выполнима.

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Идем дальше по пунктам.
    г) открыть файл /etc/enigma2/settings для чтения

    Код:
        i = open("/etc/enigma2/settings", "r")
    д) прочитать файл /etc/enigma2/settings

    Код:
        k = i.read()
    e) закрыть файл /etc/enigma2/settings

    Код:
        i.close()
    е) методом split создать список строк

    Код:
        m = k.split('\n')
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    и) удалить из этого списка те строки в которых есть подстрока Nims
    Решим эту задачу от обратного, то есть сохраним в этом списке только те строки, в которых нет подстроки (то бишь просто "слова") Nims:

    Код:
        v = []
        for x in m:
           if not x.__contains__('Nims'):
              v.append(x)
    То бишь последний код означает:
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    к) Список с оставшимися строками с помощью метода join вновь объединить в строку

    Код:
        z = '\n'.join(v)
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    л) методом простой конкатенации соединить строки полученные из файлов /etc/enigma2/settings и /tmp/mysettings и записать новый файл settings

    Код:
        new = z + b
    Это есть простая конкатенация, то есть сложение двух строк, получаем одну объединенную строку.
    Создаем в папке темп новый файл settings (с правами на запись) и пишем туда полученное:

    Код:
        s = open("/tmp/settings", "w")
        s.write(new)
    Закрываем файл (действие обязательное)

    Код:
        s.close()
    Вот весь код второго этапа (восстановление настроек)

    Код:
        a = open("/tmp/mysettings", "r")
        b = a.read()
        a.close()
        i = open("/etc/enigma2/settings", "r")
        k = i.read()
        i.close()
        m = k.split('\n')
        v = []
        for x in m:
           if not x.__contains__('Nims'):
              v.append(x)
        z = '\n'.join(v)
        new = z + b
        s = open("/tmp/settings", "w")
        s.write(new)
        s.close()
    Или так на экране в програмке Geany:
    [​IMG]

    Чтобы проверить, не ошиблись ли где, снова открываем телнет в програмке DCC, запускаем питон на нем, дав команду


    Код:
        python

    И строчку за строчкой вводим написанный нами код.

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Все нормально, файл сеттингс в папке темп создался (проверяйте сами)



    Остается последнее, собственно применить эти настройки, то есть в нашем случае просто заменить файл settings находящийся в папке /etc/enigma2/, другим файлом settings созданным нами в папке темп.
    Здесь есть вот какой нюанс.
    Этот файл (/etc/enigma2/settings) является системным, в котором имидж хранить свои настройки, соответственно любые записи которые мы туда впишем, имидж при перезагрузке перепишет на свои.
    Чтобы этого не произошло, нужно остановит имидж, перезаписать этот файл и вновь запустит. Тогда имидж будет думать, что сам сохранил эти настройки при предыдущем отключении и применит эти настройки.
    Но так как, если мы остановим имидж, остановится естественно и плагин наш, задачу замены этого файла мы будем решат скриптом, естественно заложив в плагин запуск этого скрипта в нужный момент. Код запуска этого скрипта (в нужный момент) напишем, когда будем оформлять интерфейс плагина (код очень простой).
    А сам скрипт тоже простой, это уже обычная командная строка.......

    Код:
        #!/bin/sh
        echo ""
        echo ""
        init 4
        rm -rf /etc/enigma2/settings
        cp /tmp/settings /etc/enigma2/settings
        rm -rf /tmp/settings
        init 3
        exit 0
    Это простые команды удаления и копирования. Можно добавить еще вывод сообщений в консоль (хотя пользователи ее не читают)....

    Код:
        #!/bin/sh
        echo ""
        echo ""
        echo "Restoring settings"
        echo ""
        echo ""
        echo "GUI will restart now!"
        echo ""
        echo ""
        init 4
        rm -rf /etc/enigma2/settings
        cp /tmp/settings /etc/enigma2/settings
        rm -rf /tmp/settings
        init 3
        exit 0
    Этот скрипт положим наверно вовнутрь плагина и плагин запустит его в нужный момент.

    Собственно, непосредственно с самой задачей в питоне мы справились.
    Осталось оформить все это в виде плагина....

    Далее будем именно это рассматривать...

    Продолжение следует......
     
  4. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    Если изучали питон, знаете, что графический интерфейс создается к примеру с помощью модуля Tkinter.
    Так как мы перед собой ставим задачу научится писАть или хотя бы править плагины на имидже enigma2, сразу скажу, забудьте про Tkinter.
    Нам врядь ли понадобится напрямую работать с этим модулем.
    Дело в том, что разработчики enigma2 уже позаботились о графическом интерфейсе для плагинов в том числе.
    В имидже уже имеются готовые модули окон (Screen), которые достаточно в плагине только импортировать.
    Эти окна-Screen находятся в имидже по пути /usr/lib/enigma2/python/Screen/.
    К примеру, если вы разработаете плагин-плеер для какого-нибудь видеоформата, можете использовать любой из этих окон-screen
    Infobar
    DVD

    Также известно, что вообще питон очень богат на готовые библиотечные модули, которыми просто нужно пользоваться, а кроме того еще и изучая enigma2,
    мы узнаем что в имидже по пути /usr/lib/enigma2/python/Components/ находятся также немало модулей, которыми тоже нужно пользоваться.

    Ну вот и попытаемся написать интерфейс плагина с использованием всего этого.

    Снова подготовим програмку Geany для написания кода.
    Тот код, который мы уже написали относится к двум событиям в плагине, то есть две части написанного нами кода это есть два события.
    Первое событие назовем - Сохранение текущих настроек тюнера (тюнеров).
    Второе событие - Применение ранее сохраненных настроек тюнера (тюнеров).


    Нам остается написать весь остальной код для окончательного оформления плагина.
    Порядок действий нашей дальнейшей работы:
    1. Определить какое окно-Screen нам нужно импортировать и написать инструкцию __init__ со всем необходимыми виджетами,
    с нужным акшинмапом и так далее.
    2. Затем сразу определится с разработкой главного окна в xml.
    3. Разработать меню и обработать события элементов меню и кнопок.
    4. При необходимости написать (воспользовавшись готовым шаблоном) языковую часть плагина с файлами .po и .mo
    5. Последний штрих - выбрать в каком меню имиджа нужно отобразить плагин и написать код запуска.......


    По порядку.
    1. Определить какое окно-Screen нам нужно импортировать и написать инструкцию __init__ со всем необходимыми виджетами.

    Откроем новый файл снова по шаблону main.py и не забываем про униксовые окончания строк.
    Очищаем окно Geany от шаблонного кода и приступаем.

    Как уже сказал, окно будем просто импортировать из готового /usr/lib/enigma2/python/Screen/.
    Нам в данном случае подходит окно-screen собственно под названием Screen.
    Значит первой строкой нашего кода будет импорт этого модуля

    Код:
        from Screens.Screen import Screen
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Здесь нужно разьяснение. Более продвинувшись в питоне и открыв в имидже этот файл Screen, сами разберетесь в его коде, а пока скажу вот что.
    В этом файле разработано скажем "эфемерное" окно. Там нет ни размеров, и естественно ни виджетов-элементов....
    Все это мы сами напишем.
    Импортировав Screen мы тутже его применяем.

    То есть сперва-наперва создаем класс, придумав название, в нашем случае скажем NimSettings и передаем ему в качестве аргумента (вернее как суперкласс) этот самый Screen:

    Код:
        class NimSettings(Screen):
    Так как при вызове этого класса (читай - главного окна плагина), сразу же мы должны лицезреть не пустое окно-Screen,
    то есть требуется гарантировать чтобы окно (то бишь класс) сразу содержал такие атрибуты как скажем меню, кнопки, название окна,
    создаем инструкцию-функцию с помощью метода под названием __init__ (название встроенное, а тело метода создаем сами).
    Не знаю, сумел ли я объяснить (я то сам тоже не сразу понял), но методом __init__ следует пользоваться если необходимо, чтобы интерпретатор
    автоматически вызывал метод при вызове самого класса, без необходимости вызова по имени метода.....

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Вот и начнем конструировать первую обязательную в нашем случае функцию (__init__) для класса-окна (далее для правильного восприятия и понимания буду писать сразу весь код сначала):

    Код:
        class NimSettings(Screen):
            def __init__(self, session):
    Почему в скобках у метода __init__ мы записали два аргумента self и session?
    self - атрибут обязательный, означающий экземпляр объекта. То есть интерпретатор при вызове объекта создает его экземпляр, потому self обязателен.
    Может немного сложно, но понимание роли self имеет ключевое значение.
    А session, это атрибут ожидаемый от вызывающей функции....

    Добавляем из суперкласса (то есть из импортируемого Screen) вызываем метод __init__ (обязательно):

    Код:
        class NimSettings(Screen):
            def __init__(self, session):
               Screen.__init__(self, session)
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Продолжаем оформлять наш метод __init__ и приступаем непосредственно к элементам самого окна.
    Здесь опять нужно отступление.
    Смотрите вот предыдущую строчку мы совершенно не зря вписали, не зря импортировали инициализацию модуля Screen из
    одноименного файла в папке Screens. Потому что далее будем использовать атрибуты-методы из него, а также из модулей, который уже импортирует он.
    Например из компонента GUISkin находящегося в папке Components уже.
    Далее буду просто упоминать из какого модуля какой метод.....

    Вот воспользуемся методом setTitle, который разработан уже в компоненте GUISkin (откройте его и поизучайте), чтобы сконструировать
    название окна меню, передав этому методу в качестве аргумента непосредственно придуманное нами название окна в виде строки:

    Код:
        self.setTitle("Plugin for restoring tuner settings")
    И тут же изменим эту строчку, чтобы это самое название переводилось имиджем на русский.
    Что нужно сделать для этого. Нужно импортировать gettext (сделаем потом) и указать ему, что строку нужно переводить, для
    этого достаточно взять строку в скобки и перед скобками вставить нижнее подчеркивание, то есть вот так

    Код:
        self.setTitle(_("Plugin for restoring tuner settings"))
    Весь код сначала класса

    Код:
        class NimSettings(Screen):
            def __init__(self, session):
               Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
    Правда если действительно изучите GUISkin, то увидите что также совершенно равноценно этому будет и простой виджет

    Код:
        self["Title"] = StaticText(_("Plugin for restoring tuner settings"))
    И нужно будет создать виджет с сурчом "Title", и это действительно тоже самое.....
    Кстати вот таким макаром self["Title"] - добавление к self строки в квадратных скобках, создаются виджеты в методе __init__

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Так, название самого окна меню создали. Теперь подумаем, что еще будет в нашем меню.
    а) В нашем случае нужно не конфиг-меню (то есть с ответами да-нет, с выбором вариантов), а простое меню из двух пунктов.
    Первый будет "Сохранение настроек тюнеров",
    Второй будет "Восстановление настроек тюнеров".
    б) И кроме меню, наверно нужен еще простой текстовый виджет, где будет написано "Нажмите ОК для выбора действия", ведь всегда найдутся непонятливые,
    что для выбора нужного пункта на нем просто нужно нажать ОК.
    в) Ну и одна наверно только кнопка - красная с надписью "Выход" или "Назад", кому как нравится.


    Все это конечно внутри метода __init__
    Меню создадим тоже в виде виджета. И мы знаем что наше меню это список.
    Поэтому снова воспользуемся готовым модулем List из папки Sources имиджа, импортируем этот модуль (в самом начале кода)

    Код:
        from Components.Sources.List import List
    В методе __init__ создаем для начала пустой список (назвав его как хотим):

    Код:
        nashemenu = []
    И далее создаем виджет меню, передав модулю List этот список (который затем заполним) в качестве аргумента:

    Код:
        self["menu"] = List(nashemenu)
    И весь код сначала класса

    Код:
        class NimSettings(Screen):
            def __init__(self, session):
               Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
    Следующий б) текстовый виджет
    Для вывода простого текста воспользуемся модулем Label из компонентов (для интереса изучайте в папке /usr/lib/enigma2/python/Components/Label)
    Как обычно импортируем его в начале кода

    Код:
        from Components.Label import Label
    Кстати на этот момент весь наш импорт

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.Label import Label
    И пишем код виджета с возможностью перевода

    Код:
        self["text"] = Label(_("Press OK for action"))
    И создаем красную кнопку, также виджетом

    Код:
        self["red_key"] = Label(_("Close"))
    И весь код сначала класса

    Код:
        class NimSettings(Screen):
            def __init__(self, session):
               Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
    Нужно также создать акшинмап, то есть функцию для кеймапа, то бишь действия при нажатии кнопок.
    Для этого перво-наперво импортируем модуль ActionMap из компонентов

    Код:
        from Components.ActionMap import ActionMap
    А сам код сначала напишу, потом поясню.

    Код:
        self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
    Итак здесь self["shortcuts"] понятно название, но shortcuts произвольно, можете назвать как хотите, главное уже то, что справа от знака присваивания =
    В качестве аргументов импортируемому модулю ActionMap передаются, сначала названия контекстов кеймапа имиджа.
    Ну то есть в данном случае это два контекста "ShortcutActions", "WizardActions". Их мы берем из файла кеймап.хмл имиджа. Они там прописаны как
    map context=
    Надеюсь обьяснять где находится кеймап.хмл не нужно.
    А вот второй аргумент, это словарный, справа названия кнопок из тех самых двух контекстов, но нужные только нам, а справа названия функций (сами придумаем), а тело самых этих функций,
    то есть обработчики событий напишем впоследствии дальше......

    И последнее в методе __init__ собственно должны написать название инструкции по созданию списка меню, который у нас пока пуст как помните....
    Название также произольное....
    Тело функции допишем впоследствии.

    Код:
        self.createList()
    Итого пока весь импорт и весь класс

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.Label import Label
    Код:
        class NimSettings(Screen):
            def __init__(self, session):
               Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
              self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
              self.createList()
    aabb8c883d5f.png

    Вообще-то в самом начале класса должен быть хмловский код окна-Screen, и затем уже метод __init__, но так как этот хмловский код без этого метода не создашь,
    порядок действий выбран таков.

    Далее, с помощью уже не питона, а хмл создадим само окно.
    А затем закончим код класса, разработкой функций cancel, action, createList...

    И в самом конце оформим вызов плагина.......

    На сегодня все пока.
    Думаю, поняли из вышенаписанного каждую строчку кода, начиная - почему классу в качестве аргумента передали модуль Screen, заканчивая - как создавать акшинмап.....

    Продолжение следует.......
     
  5. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    Продолжаем.

    Напоминаю полученный на этот момент код плагина

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.Label import Label
    
    
        class NimSettings(Screen):
         def __init__(self, session):
         Screen.__init__(self, session)
         self.setTitle(_("Plugin for restoring tuner settings"))
         nashemenu = []
         self["menu"] = List(nashemenu)
         self["text"] = Label(_("Press OK for action"))
         self["red_key"] = Label(_("Close"))
         self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
         {
         "cancel": self.cancel,
         "back": self.cancel,
         "red": self.cancel,
         "ok": self.action,
         })
         self.createList()
    Ну и кроме того, коды по сохранению настроек и применению настроек, которые мы в самом начале разработали, держим пока в уме (они сохранены на диске)......

    Из этого кода например из части действий нажатия кнопок думаю вы поняли, что вот эти три ключа "cancel", "back", "red" это разновидности красной кнопки в разных имиджах и в разных ресах. В некоторых ресах есть еще и кнопка "назад" например........

    Разработаем конструкцию самого окна-меню плагина.
    В принципе это уже не питон.
    Это уже xml.
    То есть нужно разработать окно-Screen этого плагина для скина.
    Это уже чисто построение каркаса в скиностроении.
    Поэтому подробно показывать как построил каркас на основе уже готовых виджетов в инструкции __init__ не буду. Как строить каркасы в скинах читайте в другом FAQ здесь Пожалуйста войдите или зарегистируйтесь
    Поясню только основное.
    Каркас для скина размещается в самом начале класса, то есть сразу за заглавием класса
    class NimSettings(Screen):
    и синтаксис такой: переменной skin присваивается весь текст-каркас этого окна.
    Ну то есть
    skin = каркас_скина
    Только так как каркас скина это многострочный текст, по правилам питона он берется в тройные кавычки
    """каркас_скина_в_виде_многострочного_текста"""

    Вобщем создаем каркас окна-Screen с названием точно таким же как название класса (это обязательно), title можем пустой оставить или написать что угодно(все равно применится self.setTitle), атрибуты (размер скина, расположение на экране) придумываем как нам нужно, и добавляем все виджеты (красную кнопку, виджет меню, виджет текст) из инструкции __init__
    Я обычно строю каркас плагина в окне e2skinner.
    Беру какой-нибудь похожий каркас и меняю там виджеты на свои.
    Здесь главное не ошибится например в сурчах.
    Вот к примеру кнопку можно оформить как через компонент Label, то есть так (как в нашем примере):

    Код:
        self["red_key"] = Label(_("Close"))
    а можно и через сурч StaticText, то есть так:

    Код:
        self["red_key"] = StaticText(_("Close"))
    В первом случае сам виджет в каркасе будет выглядеть так, просто через name =...:

    Код:
        <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
    А во втором случае это будет через source и через рендер, так:

    Код:
        <widget source="red_key" render="Label" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
    Тонкости совсем несложные, думаю быстро разберетесь.
    А меню само будем строить через рендер ListBox и с помощью TemplatedMultiContent
    Вобщем итого получаем такой код вместе с хмл, то есть с каркасом:

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.Label import Label
    
    
        class NimSettings(Screen):
         skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
         def __init__(self, session):
         Screen.__init__(self, session)
         self.setTitle(_("Plugin for restoring tuner settings"))
         nashemenu = []
         self["menu"] = List(nashemenu)
         self["text"] = Label(_("Press OK for action"))
         self["red_key"] = Label(_("Close"))
         self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
         {
         "cancel": self.cancel,
         "back": self.cancel,
         "red": self.cancel,
         "ok": self.action,
         })
         self.createList()
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Пришла очередь разработать функции self.cancel, self.action, self.createList.
    Сначала createList, то есть меню плагина, оно пока пустое.
    С помощью встроенного атрибута append добавляем в этот пустой список два пункта, а затем
    этот уже непустой список передаем в качестве аргумента методу setList модуля List (чтобы понять, изучите
    импортированный нами List из папки Sources), то есть так:

    Код:
        def createList(self):
         nashemenu = []
         nashemenu.append((_("Save tuner settings"), "save"))
         nashemenu.append((_("Restore tuner settings"), "restore"))
         self["menu"].setList(nashemenu)
    Разберем каждую строчку снова.
    nashemenu = [] # Просто присвоили (снова) пустой список переменной nashemenu
    nashemenu.append((_("Save tuner settings"), "save")) # добавили один элемент списка, то есть попросту первую строчку меню.

    Здесь нужно понять вот что. Встроенный метод списков append добавить в список первый атрибут, который мы ему передали,
    то есть вот это _("Save tuner settings"), а второй атрибут в данном случае будет просто индексом.
    Этот второй атрибут-индекс, в данном случае наша питоновская импровизация, для того чтобы впоследствии
    именно по нему, по второму атрибуту-индексу определить собственно в какой строчке нажал кнопку ОК пользователь.
    Это именно питоновская импровизация, вы же можете пойти другим путем и определить это дело по-другому,
    использую другие богатые возможности питона.
    Понимание этого потихоньку придет к вам, если будете вникать в смысл, а не тупо копировать.....
    Пока же просто имейте ввиду, что именно по этим двум "словам"-индексам "save" и "restore" мы будем определять,
    в какой строчке пользователь нажал кнопку ОК (то есть хочет сохранить настройки или наоборот применить ранее сохраненные).

    self["menu"].setList(nashemenu) # Это опять-таки применение метода setList из импортированного модуля List

    Чтобы не путаться, пишем снова весь код, который получили

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.Label import Label
    
    
        class NimSettings(Screen):
         skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
         def __init__(self, session):
         Screen.__init__(self, session)
         self.setTitle(_("Plugin for restoring tuner settings"))
         nashemenu = []
         self["menu"] = List(nashemenu)
         self["text"] = Label(_("Press OK for action"))
         self["red_key"] = Label(_("Close"))
         self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
         {
         "cancel": self.cancel,
         "back": self.cancel,
         "red": self.cancel,
         "ok": self.action,
         })
         self.createList()
       
         def createList(self):
         nashemenu = []
         nashemenu.append((_("Save tuner settings"), "save"))
         nashemenu.append((_("Restore tuner settings"), "restore"))
         self["menu"].setList(nashemenu)
    Следующая функция, это действие, происходящее при нажатии красной кнопки (или exit), то есть функция self.cancel , так как это всего лишь закрытие окна, это самая простая функция:

    Код:
        def cancel(self):
         self.close()
    Тут ничего объяснять не надо, close он и в африке close....

    Осталась главная функция - это действие, которое будет происходит при нажатии кнопки ОК, то есть функция self.action

    Продолжение следует.....
     
  6. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    Ну и теперь функция self.action, то есть обработчик событий при нажатии кнопки ОК.
    Вот здесь наконец нам понадобится тот самый код, который мы в самом начале разработали.
    Это главная функция, которая и составляет смысл нашего плагина.
    В качестве аргумента функции, кроме собственно self, создадим аргумент для вычисления выбранного пункта меню и назовем по смыслу currentSelect и по умолчанию присвоим этому аргументу значение "Ложь", то есть так:

    Код:
        def action(self, currentSelect = None):
    Попробую пояснить.
    Если просто создать аргумент (не приравнивая его к значению "Ложь") и далее проверять какая эта строка, при первом прохождении функции (то есть при первой проверке, на каком пункте курсор) полученное значение будет присвоино этому аргументу и при следующей проверке(если вы это сделали до перезагрузки) поведение будет непредсказуемым, то есть окажется что курсор не на той строчке или на двух строчках сразу (так как предыдущее значение-то не сбросилось, а сохранилось).
    Именно чтобы избежать этого, каждый раз при начале проверки, значение переменной-аргумента currentSelect обнуляется и сам код именно поэтому начнем с условия "Если нет выбора"- if currentSelect is None:


    Код:
           def action(self, currentSelect = None):
              if currentSelect is None:
    и высчитываем спокойно текущую строчку:

    Код:
        currentSelect = self["menu"].getCurrent()[1]
    Тут очень просто self["menu"] это как известно из инструкции __init__ и есть импортированный модуль List, а у этого модуля есть метод для вычисления текущей строки getCurrent() (проверяйте это, открыв и изучив сам модуль List). А [1] - это как мы знаем по питону, простое "выдергивание" первого индекса из элемента. Также по питону знаем, что индексация начинается с нуля, то бишь элемент номер 1 - это на самом деле второй элемент, так как первым является нулевой элемент. То есть вышенаписанный код находит, что текущая строка это -(_("Save tuner settings"), "save")
    Здесь индекс 0 - _("Save tuner settings"), а индекс 1 - "save".
    Так как мы запросили индекс 1 (а не индекс 0), то есть [1], мы получаем строку "save"

    Соответственно если курсор стоит на второй строчке, к нам вернется строка "restore".
    Все логично и очень просто.

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Вот полученный код этой функции пока:

    Код:
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]

    Далее еще проще. Если полученное значение "save", то понятное дело это означает что пользователь потребовал сохранения настроек(нажал кнопку ОК на пункте "Сохранить настройки").
    Соответственно написав if selectEntry is "save": за ним пишем тот самый код (из начала урока) по сохранению настроек.
    Но прежде чем написать это, рассмотрим варианты.
    К примеру ведь может оказаться что пользователь уже сохранял настройки (в заранее указанную папку).
    Либо пользователь случайно нажал кнопку ОК в этой строчке, а на самом деле хотел нажать на другой строчке. Конечно можем на все эти обстоятельства плюнуть и всегда сохранять настройки в файл (если уже есть перезапишется). Но это неумно и не защищает пользователя от ошибочного нажатия кнопки ОК.
    Поэтому напишем код таким образом (просто добавится еще одно вложение - а существует ли файл уже):
    if currentSelect is "save":
    if fileExists(путь_к_файлу + 'имя_файла'):


    В коде появилось что-то новое. Разберем по порядку.
    Здесь воспользовались методом fileExists, который мы не будем сами разрабатывать, а как всегда "одолжим" у существующего модуля в имидже Directories (из папки Tools).
    Если интересно узнать (должно быть интересно, если хотим расти), как работает этот метод, идем по пути /usr/lib/enigma2/python/Tools/ и открывая модуль
    Directories - изучаем там содержимое метода. Этот метод просто проверяет наличие или отстутвие файла, который ему передан в качестве аргумента.
    Конечно обязательно импортируем модуль в начале кода плагина:

    Код:
        from Tools.Directories fileExists
    Теперь самое время нам разобраться с этим
    путь_к_файлу + 'имя_файла'
    Определимся куда будем сохранять файл настроек. Для этого сначала придумаем название нашему плагу. Пусть будет по смыслу так - TunerSettingsRestore.
    И создадим внутри плагина папку backup и сохраним туда настройки.
    Плагин наш будет находится в папке Extensions, сам файл пусть как первоначально придумали назовем mysettings, соответственно получаем:

    Код:
        f currentSelect is "save":
           if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
    Или весь код сначала с учетом импорта:

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.ActionMap import ActionMap
        from Components.Label import Label
        from Tools.Directories fileExists
    
    
        class NimSettings(Screen):
           skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
           def __init__(self, session):
              Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
              self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
              self.createList()
    
           def createList(self):
              nashemenu = []
              nashemenu.append((_("Save tuner settings"), "save"))
              nashemenu.append((_("Restore tuner settings"), "restore"))
              self["menu"].setList(nashemenu)
    
           def cancel(self):
              self.close()
    
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
    А код функции action пока приняла этот вид:

    Код:
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):

    Итак, что же делать, если файл уже существует?
    Надо предоставить пользователю выбор, либо вернуться в меню плагина, либо, если он хочет, перезаписать файл.
    Для этого вызовем окно выбора с помощью встроенного метода openWithCallback и существующего в имидже модуля MessageBox.
    Для начала выполним необходимый импорт нужного нам модуля.

    Код:
        from Screens.MessageBox import MessageBox
    Первым аргументом у метода openWithCallback идет название функции (назовем saveset), которая будет выполнена при положительном ответе (при отрицательном просто выход из окна назад). а вторым аргументом мы ему передадим импортированный MessageBox, а далее уже аргументы соответственно самого MessageBox.
    Открываем модуль MessageBox по пути /usr/lib/enigma2/python/Screens/ и из его метода __init__ узнаем какие атрибуты ему нужны. Для наглядности приведу эти аргументы:
    (self, session, text, type = TYPE_YESNO, timeout = -1, close_on_any_key = False, default = True, enable_input = True, msgBoxID = None, picon = None, simple = False, list = [])
    Как видим ему обязательно нужно передать только сам собственно text, а остальные аргументы имеют значения по умолчанию и если мы не собираемся их менять, то можно их вообще не указывать.
    По порядку:

    а) Как text передадим _("Backup file is already exists!\nDo you want rewrite backup file?"), думаю понятно (знак \n - просто перенос строки)
    б) type = TYPE_YESNO, нас вполне устраивает, нам нужен именно тип TYPE_YESNO, значит пропускаем.
    в) timeout = -1, а вот это непорядок (-1 означает нет), ставим таймаут 6 секунд timeout = 6,
    г) close_on_any_key = False, устраивает, мы не собираемся закрыть окно любой кнопкой
    д) default = True, вот здесь надо подумать, True здесь означает, что курсор будет на строчке - "да", мы
    поставим на строчку "нет" на всякий пожарный, чтобы уберечь пользователя от ошибок, то есть default = False,
    е) остальные атрибуты вообще не трогаем и оставляем по умолчанию.

    И получаем такой вызов окна:

    Код:
        self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
    
    Обратите внимание, мы вообще не указали тип окна YESNO или другой, просто в данном случае нам нужен именно YESNO, а он применется по умолчанию, если нужен другой тип окна, тогда нужно указывать...
    Дополнительную нужную функцию self.saveset разработаем после того, как закончим с функцией action.

    То есть понятно, выведя окно сообщения, мы спросили у пользователя, что настройки бекапа уже существуют, нужно ли их перезаписать, и по умолчанию поставили курсор на ответе "нет", на всякий случай.....
    При ответе "да" будет запущена функция self.saveset, которая перезапишет настройки (разработаем впоследствии), при ответе "нет" возврат в предыдущее окно.

    Полученный код функции:

    Код:
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
    Далее код, если файл не существует. Здесь за условием else: (то бишь если файл не существует) запишем какраз таки от самый код сохранения настроек, который мы разрабатывали самом начале, то есть так:

    Код:
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
                    else:
                       a = open("/etc/enigma2/settings", "r")
                       b = a.read()
                       a.close()
                       c = b.split('\n')
                       d = []
                       for x in c:
                          if x.__contains__('Nims'):
                             d.append(x)
                       e = '\n'.join(d)
                       f = e + '\n'
                       g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings","w")
                       g.write(f)
                       g.close()

    Только естественно теперь указали путь сохранения настроек туда, куда нам надо, то есть /usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/

    И далее, восстановление настроек, то есть если нажата ОК в строчке с сохранением настроек, то есть с индексом "restore", то есть:

    Код:
        elif currentSelect is "restore":
    Здесь тоже нужно придумать часть (которую программисты называют "защитой от дураков"), например для случая, если собственно сохраненного файла настроек-то нет, а также, в случае если даже и есть такие настройки, еще раз спросить пользователя, дейчтвительно ли хочет....
    Как вы понимаете, если файла настроек нет, то и нечего восстанавливать.
    Поэтому проверим наличие файла и при отсутствии оного, просто покажем сообщение, что этого файла нет и вернемся в предыдущее окно.
    Воспользуемся другим встроенным методом текущей сессии open и также тем же модулем MessageBox.
    Метод open текущей сессии требует только аргумент с названием окна (MessageBox) и далее также управление передается собственно модулю MessageBox.
    И так как на этот раз нам нужен другой тип окна TYPE_INFO(а не который по умолчанию), на этот раз явно укажем этот тип. И получим такой код вызова окна сообщения:

    Код:
                 elif currentSelect is "restore":
                    if not fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.open(MessageBox, _("Backup file not exists!"), MessageBox.TYPE_INFO, timeout = 6)

    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Далее если тот файл есть выводим сообщение, что сейчас будет имидж остановлен (тем самым скриптом), и перезапущен с новыми настройками, но пользователь может отменить и передумать.
    То есть это будет опять метод openWithCallback, функцию, которую он запустить при положительном ответе, назовем self.restoreset, остальное вы уже по типу пройденного поймете:

    Код:
                    elif fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.restoreset, MessageBox, _("Do not be afraid, now image will be stopped and restarted for the restore tuner settings.\nRestore settings?"), timeout = 6)
    Функция self.restoreset будет находиться за пределами функции self.action, поэтому разработаем ее после.

    Итого, теперь весь полученный нами код нашего плагина на этот момент:

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.ActionMap import ActionMap
        from Components.Label import Label
        from Tools.Directories import fileExists
        from Screens.MessageBox import MessageBox
    
    
        class NimSettings(Screen):
           skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
           def __init__(self, session):
              Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
              self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
              self.createList()
    
           def createList(self):
              nashemenu = []
              nashemenu.append((_("Save tuner settings"), "save"))
              nashemenu.append((_("Restore tuner settings"), "restore"))
              self["menu"].setList(nashemenu)
    
           def cancel(self):
              self.close()
    
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
                    else:
                       a = open("/etc/enigma2/settings", "r")
                       b = a.read()
                       a.close()
                       c = b.split('\n')
                       d = []
                       for x in c:
                          if x.__contains__('Nims'):
                             d.append(x)
                       e = '\n'.join(d)
                       f = e + '\n'
                       g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings","w")
                       g.write(f)
                       g.close()
                 elif currentSelect is "restore":
                    if not fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.open(MessageBox, _("Backup file not exists!"), MessageBox.TYPE_INFO, timeout = 6)
                    elif fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.restoreset, MessageBox, _("Do not be afraid, now image will be stopped and restarted for the restore tuner settings.\nRestore settings?"), timeout = 6)
    

    Остается в классе разработать дополнительно появившиеся функции self.saveset и self.restoreset.

    Продолжение следует....
     
    leon нравится это.
  7. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    Прежде чем разработаем последние две функции класса, сначала в предыдущем коде внесем маленькое дополнение.
    В функции action, в ветке сохранения настроек(когда ранее сохраненного файла не оказалось), мы просто сохранили настройки. А нужно бы по логике вещей выдать сообщение, чтобы пользователь понял, что настройки успешно сохранились. Поэтому в той части кода, сразу за cохранением настроек исходя из логики добавим вывод сообщения, что настройки успешно сохранены, то есть вот так:

    Код:
        self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)
    И весь код приобретает вид:

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.ActionMap import ActionMap
        from Components.Label import Label
        from Tools.Directories import fileExists
        from Screens.MessageBox import MessageBox
    
    
        class NimSettings(Screen):
           skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
           def __init__(self, session):
              Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
              self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
              self.createList()
    
           def createList(self):
              nashemenu = []
              nashemenu.append((_("Save tuner settings"), "save"))
              nashemenu.append((_("Restore tuner settings"), "restore"))
              self["menu"].setList(nashemenu)
    
           def cancel(self):
              self.close()
    
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
                    else:
                       a = open("/etc/enigma2/settings", "r")
                       b = a.read()
                       a.close()
                       c = b.split('\n')
                       d = []
                       for x in c:
                          if x.__contains__('Nims'):
                             d.append(x)
                       e = '\n'.join(d)
                       f = e + '\n'
                       g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup//mysettings","w")
                       g.write(f)
                       g.close()
                       self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)
                 elif currentSelect is "restore":
                    if not fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.open(MessageBox, _("Backup file not exists!"), MessageBox.TYPE_INFO, timeout = 6)
                    elif fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.restoreset, MessageBox, _("Do not be afraid, now image will be stopped and restarted for the restore tuner settings.\nRestore settings?"), timeout = 6)
    Последние две функции нашего класса.
    self.saveset это просто сохранение настроек (после подтверждения о перезаписи существующего файла):

    Код:
           def saveset(self, answer):
              if answer is True:
                 a = open("/etc/enigma2/settings", "r")
                 b = a.read()
                 a.close()
                 c = b.split('\n')
                 d = []
                 for x in c:
                    if x.__contains__('Nims'):
                       d.append(x)
                 e = '\n'.join(d)
                 f = e + '\n'
                 g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup//mysettings","w")
                 g.write(f)
                 g.close()
                 self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)

    Это тот самый код для сохранения, уже разбирали по буквочкам. Здесь у функции присутствует аргумент answer и соответственно код выполнится, если if answer is True:, то есть при получении положительного ответа....

    И последняя функция нашего класса self.restoreset, которая будет задействована после того, как пользователь нажал ОК в пункте сохранения настроек и подтвердил затем сохранение.
    Наконец-то именно здесь применим тот код для сохранения, который разрабатывали в самом начале, вот так:

    Код:
           def restoreset(self, answer):
              if answer is True:
                 a = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings", "r")
                 b = a.read()
                 a.close()
                 i = open("/etc/enigma2/settings", "r")
                 k = i.read()
                 i.close()
                 m = k.split('\n')
                 v = []
                 for x in m:
                    if not x.__contains__('Nims'):
                       v.append(x)
                 z = '\n'.join(v)
                 new = z + b
                 s = open("/tmp/settings", "w")
                 s.write(new)
                 s.close()

    Это тот самый код, что разрабатывали в самом начале урока, за исключением пути сохранения
    файла настроек (если помните тогда мы сохраняли временно в папку темп).

    Здесь остается только добавить запуск скрипта с восстановлением настроек.
    Напомню текст скрипта:

    Код:
        #!/bin/sh
        echo ""
        echo ""
        echo "Restoring settings"
        echo ""
        echo ""
        echo "GUI will restart now!"
        echo ""
        echo ""
        init 4
        rm -rf /etc/enigma2/settings
        cp /tmp/settings /etc/enigma2/settings
        rm -rf /tmp/settings
        init 3
        exit 0 
    Сохраним этот скрипт в папку с плагином, создав для этого отдельную папку script, и назвав скажем файл скрипта 'restore_settings.sh'.

    В таком разе код запуска скрипта будет выглядеть так:

    Код:
        from Screens.Console import Console
        import os
        script = "/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/script/restore_settings.sh"
        os.chmod(script, 0755)
        self.session.open(Console, cmdlist=[script])
    Соответственно функция restoreset примет вид:

    Код:
           def restoreset(self, answer):
              if answer is True:
                 a = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings", "r")
                 b = a.read()
                 a.close()
                 i = open("/etc/enigma2/settings", "r")
                 k = i.read()
                 i.close()
                 m = k.split('\n')
                 v = []
                 for x in m:
                    if not x.__contains__('Nims'):
                       v.append(x)
                 z = '\n'.join(v)
                 new = z + b
                 s = open("/tmp/settings", "w")
                 s.write(new)
                 s.close()
                 from Screens.Console import Console
                 import os
                 script = "/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/script/restore_settings.sh"
                 os.chmod(script, 0755)
                 self.session.open(Console, cmdlist=[script])

    Код класса в плагине(то есть меню плагина, со всеми действиями с этим меню) полностью готов.
    А ведь мы создали целый модуль собственный! И причем абсолютно рабочий.....
    Вот он:

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.ActionMap import ActionMap
        from Components.Label import Label
        from Tools.Directories import fileExists
        from Screens.MessageBox import MessageBox
    
    
        class NimSettings(Screen):
           skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
           def __init__(self, session):
              Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
              self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
              self.createList()
    
           def createList(self):
              nashemenu = []
              nashemenu.append((_("Save tuner settings"), "save"))
              nashemenu.append((_("Restore tuner settings"), "restore"))
              self["menu"].setList(nashemenu)
    
           def cancel(self):
              self.close()
    
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
                    else:
                       a = open("/etc/enigma2/settings", "r")
                       b = a.read()
                       a.close()
                       c = b.split('\n')
                       d = []
                       for x in c:
                          if x.__contains__('Nims'):
                             d.append(x)
                       e = '\n'.join(d)
                       f = e + '\n'
                       g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings","w")
                       g.write(f)
                       g.close()
                       self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)
                 elif currentSelect is "restore":
                    if not fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.open(MessageBox, _("Backup file not exists!"), MessageBox.TYPE_INFO, timeout = 6)
                    elif fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.restoreset, MessageBox, _("Do not be afraid, now image will be stopped and restarted for the restore tuner settings.\nRestore settings?"), timeout = 6)
    
           def saveset(self, answer):
              if answer is True:
                 a = open("/etc/enigma2/settings", "r")
                 b = a.read()
                 a.close()
                 c = b.split('\n')
                 d = []
                 for x in c:
                    if x.__contains__('Nims'):
                       d.append(x)
                 e = '\n'.join(d)
                 f = e + '\n'
                 g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings","w")
                 g.write(f)
                 g.close()
                 self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)
    
           def restoreset(self, answer):
              if answer is True:
                 a = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings", "r")
                 b = a.read()
                 a.close()
                 i = open("/etc/enigma2/settings", "r")
                 k = i.read()
                 i.close()
                 m = k.split('\n')
                 v = []
                 for x in m:
                    if not x.__contains__('Nims'):
                       v.append(x)
                 z = '\n'.join(v)
                 new = z + b
                 s = open("/tmp/settings", "w")
                 s.write(new)
                 s.close()
                 from Screens.Console import Console
                 import os
                 script = "/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/script/restore_settings.sh"
                 os.chmod(script, 0755)
                 self.session.open(Console, cmdlist=[script])
    И осталось дело за совсем малым, написать код вызова плагина и если нужно добавить локализацию
    для менюшек и сообщений нашего плага.

    Вызов плагина. Тут очень просто.
    Тоже ничего придумывать питоновского вобщем-то не надо.
    Об этом тоже позаботились создатели имиджей.

    В самом конце нашего кода, после тела класса NimSettings создаем функцию Plugins, который в качестве аргументов будет ожидать аргументы словарного типа (то бишь **kwargs), эта функция будет возвращать модуль PluginDescriptor, аргументы для которой мы придумаем.

    Код:
        def Plugins(path, **kwargs):
           return PluginDescriptor(**kwargs)
    Сам модуль PluginDescriptor импортируем из файла Plugin.py из папки Plugins

    Код:
        from Plugins.Plugin import PluginDescriptor

    Открываем этот самый Plugin.py и изучаем какие же аргументы нужно передать этому самому модулю PluginDescriptor, для этого достаточно просмотреть аргументы его инструкции __init__
    Вот эти аргументы:

    Код:
        def __init__(self, name = "Plugin", where = [ ], description = "", icon = None, fnc = None, wakeupfnc = None, needsRestart = None, internal = False, weight = 0):
    Рассмотрим по одной.
    а) name = "Plugin" #название плагина, у нас будет name = "Restore Tuner settings"
    б) where = [ ] # в каком меню показать плагин, нам достаточно показать в списке плагинов, запишем where = [PluginDescriptor.WHERE_PLUGINMENU]
    в) description = "" # описание плагина, запишем description = "Plugin for save Tuner settings"
    г) icon = None # пропустим, так как у нас нет собственной иконки для плага
    д) fnc = None # собственно что нужно задействовать при выборе плагина, запишем функцию, которую назовем произвольно, например fnc = main
    е) остальные атрибуты оставляем по умолчанию....

    Впишем эти аргументы нами придуманные в функцию вызова плагина, добавив при необходимости скобки с нижним подчеркиванием для переводимости
    при необходимости:

    Код:
        def Plugins(path, **kwargs):
           return PluginDescriptor(name = _("Restore Tuner settings"), where = [PluginDescriptor.WHERE_PLUGINMENU], description = _("Plugin for save Tuner settings"), fnc = main)

    И до этой функции предпоследним напишем функцию main, которая просто при выборе строчки с плагином (в списке плагинов)
    и запустит наш класс NimSettings:

    Код:
        def main(session, **kwargs):
           session.open(NimSettings)
    Все. Наш плагин готов в принципе к использованию.
    Осталось только два нюанса.
    Первый.
    Если мы хотим, чтобы плагин был русскоязычным, нужно заложить в плагин локализуемость и добавить файлы локализации.
    Сильно углубляться тут в код не буду. В этом случае-то всегда можете использовать готовый шаблон, да и код достаточно прост.

    Для создания переводимости меню и сообщений плагина (кроме тех скобок и нижнего подчеркивания, что мы уже добавляли в плагин)
    нужно для начала импортировать
    Код:
        from Components.Language import language
        from Tools.Directories import resolveFilename, SCOPE_PLUGINS, SCOPE_LANGUAGE
        from os import environ, system
        import gettext
    Затем в самом начале кода плагина тоже (сразу за импортами) прописать вот такой простой код

    Код:
        lang = language.getLanguage()
        environ["LANGUAGE"] = lang[:2]
        gettext.bindtextdomain("enigma2", resolveFilename(SCOPE_LANGUAGE))
        gettext.textdomain("enigma2")
        gettext.bindtextdomain("NimSettings", "%s%s" % (resolveFilename(SCOPE_PLUGINS), "Extensions/TunerSettingsRestore/locale/"))
    
    
        def _(txt):
           t = gettext.dgettext("NimSettings", txt)
           if t == txt:
              t = gettext.gettext(txt)
           return t
    Если хотите разобрать этот код по буквочкам, изучаем gettext.py, Language.py и os.py (в имидже).
    А так понятно "NimSettings" - это так мы назовем файл локализации для плагина, который будет находится,
    как и указано в этом коде Extensions/TunerSettingsRestore/locale/

    А сам файл локализации будем создавать лучше и быстрей всего из готового шаблона.
    Создайте себе шаблон например взяв айл локали вашего другого плагина, и просто изменив название
    создавать новый файл локали.
    К файлам локали (po и mo) тоже есть свои требования, объяснять их не задача нашей темы.


    Вот теперь код, который получился у нас окончательно. Наш plugin.py

    Код:
        from Screens.Screen import Screen
        from Components.Sources.List import List
        from Components.ActionMap import ActionMap
        from Components.Label import Label
        from Tools.Directories import fileExists, resolveFilename, SCOPE_PLUGINS, SCOPE_LANGUAGE
        from Screens.MessageBox import MessageBox
        from Components.Language import language
        from Plugins.Plugin import PluginDescriptor
        from os import environ, system
        import gettext
    
    
        lang = language.getLanguage()
        environ["LANGUAGE"] = lang[:2]
        gettext.bindtextdomain("enigma2", resolveFilename(SCOPE_LANGUAGE))
        gettext.textdomain("enigma2")
        gettext.bindtextdomain("NimSettings", "%s%s" % (resolveFilename(SCOPE_PLUGINS), "Extensions/TunerSettingsRestore/locale/"))
    
    
        def _(txt):
           t = gettext.dgettext("NimSettings", txt)
           if t == txt:
              t = gettext.gettext(txt)
           return t
    
    
        class NimSettings(Screen):
           skin = """
        <screen name="NimSettings" position="center,center" size="595,450" title="Backup settings for current skin">
          <widget name="text" position="69,339" size="500,30" foregroundColor="#00009a00" font="Regular;22" halign="right" transparent="1" />
          <ePixmap position="215,438" zPosition="1" size="165,2" pixmap="/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/images/red.png" alphatest="blend" />
          <widget name="red_key" position="215,408" zPosition="2" size="165,30" font="Regular; 20" halign="center" valign="center" backgroundColor="#41000000" foregroundColor="#00dddddd" transparent="1" />
          <widget source="menu" render="Listbox" position="15,10" size="570,100" scrollbarMode="showOnDemand" transparent="1">
            <convert type="TemplatedMultiContent">
              {"template": [
                MultiContentEntryText(pos = (15, 5), size = (570, 30), font=0, flags = RT_HALIGN_LEFT, text = 0)
                ],
                "fonts": [gFont("Regular", 23)],
                "itemHeight": 40
              }
            </convert>
          </widget>
        </screen>"""
    
           def __init__(self, session):
              Screen.__init__(self, session)
              self.setTitle(_("Plugin for restoring tuner settings"))
              nashemenu = []
              self["menu"] = List(nashemenu)
              self["text"] = Label(_("Press OK for action"))
              self["red_key"] = Label(_("Close"))
              self["shortcuts"] = ActionMap(["ShortcutActions", "WizardActions"],
              {
                 "cancel": self.cancel,
                 "back": self.cancel,
                 "red": self.cancel,
                 "ok": self.action,
                 })
              self.createList()
    
           def createList(self):
              nashemenu = []
              nashemenu.append((_("Save tuner settings"), "save"))
              nashemenu.append((_("Restore tuner settings"), "restore"))
              self["menu"].setList(nashemenu)
    
           def cancel(self):
              self.close()
    
           def action(self, currentSelect = None):
              if currentSelect is None:
                 currentSelect = self["menu"].getCurrent()[1]
                 if currentSelect is "save":
                    if fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.saveset, MessageBox,_("Backup file is already exists!\nDo you want rewrite backup file?"), timeout = 6, default = False)
                    else:
                       a = open("/etc/enigma2/settings", "r")
                       b = a.read()
                       a.close()
                       c = b.split('\n')
                       d = []
                       for x in c:
                          if x.__contains__('Nims'):
                             d.append(x)
                       e = '\n'.join(d)
                       f = e + '\n'
                       g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings","w")
                       g.write(f)
                       g.close()
                       self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)
                 elif currentSelect is "restore":
                    if not fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.open(MessageBox, _("Backup file not exists!"), MessageBox.TYPE_INFO, timeout = 6)
                    elif fileExists('/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/' + 'mysettings'):
                       self.session.openWithCallback(self.restoreset, MessageBox, _("Do not be afraid, now image will be stopped and restarted for the restore tuner settings.\nRestore settings?"), timeout = 6)
    
           def saveset(self, answer):
              if answer is True:
                 a = open("/etc/enigma2/settings", "r")
                 b = a.read()
                 a.close()
                 c = b.split('\n')
                 d = []
                 for x in c:
                    if x.__contains__('Nims'):
                       d.append(x)
                 e = '\n'.join(d)
                 f = e + '\n'
                 g = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings","w")
                 g.write(f)
                 g.close()
                 self.session.open(MessageBox, _("Settings successfully saved!"), MessageBox.TYPE_INFO, timeout = 6)
    
           def restoreset(self, answer):
              if answer is True:
                 a = open("/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/backup/mysettings", "r")
                 b = a.read()
                 a.close()
                 i = open("/etc/enigma2/settings", "r")
                 k = i.read()
                 i.close()
                 m = k.split('\n')
                 v = []
                 for x in m:
                    if not x.__contains__('Nims'):
                       v.append(x)
                 z = '\n'.join(v)
                 new = z + b
                 s = open("/tmp/settings", "w")
                 s.write(new)
                 s.close()
                 from Screens.Console import Console
                 import os
                 script = "/usr/lib/enigma2/python/Plugins/Extensions/TunerSettingsRestore/script/restore_settings.sh"
                 os.chmod(script, 0755)
                 self.session.open(Console, cmdlist=[script])
    
        def main(session, **kwargs):
           session.open(NimSettings)
    
        def Plugins(path, **kwargs):
           return PluginDescriptor(name = _("Restore Tuner settings"), where = [PluginDescriptor.WHERE_PLUGINMENU], description = _("Plugin for save Tuner settings"), fnc = main)
    Чтобы увидеть скрытый код необходимо зарегистрироваться.
    Файл локали тоже скажем создали.
    Остался последний нюанс.
    Файл __init__.py в папке плагина.
    Смысл этого файла:
    а) присутствие этого файла нужно, если из этой директории (то есть из папки плагина) потребуется импорт
    какого-то модуля. Например, если ваш плагин будет состоять из нескольких файлов-модулей (пакетов).
    б) если заведомо нужно объявлять атрибуты для этой директории. В нашем случае не нужно.

    Потому оставляем просто пустой (все равно униксовый должен быть, хоть и пустой) файл __init__.py


    Вот теперь все.
    Наш плагин полностью готов.
    Наверно вместе со мной создавали параллельно файлы плагина?

    Вот такая папка с файлами плагина должна была получиться, которую в архиве приложил.
    Забрасываем эту папку в имидж по пути

    /usr/lib/enigma2/python/Plugins/Extensions/

    Перегружаемся.
    И заходя в меню списка плагинов, видим наш плагин
    61e211df44fc.jpg
    f3313b2bf83e.jpg 76c8f42c1f45.jpg
    987eccccdcdc.jpg
    Поздравляю
    Вы вместе со мной написали свой первый рабочий плагин!

    А если серъезно, достиг ли я цели, которую поставил перед написанием плагина публично?
    Целью была показать новичкам изучающим питон, как нудную зубрежку питона можно и нужно применить в плагиностроении.
    Если бы например когда я только начал изучать питон, набрел бы на такой ФАК, как бы я сэкономил свое время....
    А может кого-то и подстегну к плагиностроению и в нашем полку плагинописателей прибудет возможно?

    Надеюсь кому-то помог.....
     
    Opalowich, Васильевич, gurasik431 и ещё 1-му нравится это.
  8. gurasik431
    Offline

    gurasik431 Гуру в квадрате

    Регистрация:
    30-05-2015
    Сообщения:
    2.522
    Симпатии:
    2.695
    Баллы:
    113
    Пол:
    Мужской
    Действительно, прочитать и сразу ;) забыть:p
     
  9. TopicStarter Overlay
    kil1970
    Offline

    kil1970 АДМИНОСНОВАТЕЛЬ ФОРУМА Команда форума

    Регистрация:
    04-05-2015
    Сообщения:
    3.776
    Симпатии:
    4.661
    Баллы:
    113
    Пол:
    Мужской
    Адрес:
    Ukraine
    ICQ:
    112703277
    Скайп:
    kil1970
    Сообщение администратора
    ну я приблизительно такие ответы и ждал ..........
    кому это интресно думаю напишет иное
     
    fansat, Васильевич и leon нравится это.
  10. fansat
    Offline

    fansat Любитель

    Регистрация:
    22-01-2018
    Сообщения:
    33
    Симпатии:
    19
    Баллы:
    8
    Пол:
    Мужской
    Уважаемый kil1970.
    Отличная работа. Продолжайте.
    Спасибо за Ваш труд.
     

Просматривают тему (Пользователей: 0, Гостей: 0)