Compare commits

..

18 Commits

Author SHA1 Message Date
DYefremov
8d485a9993 minor style corrections 2022-02-27 09:03:39 +03:00
Víctor Pont
1721567731 Spanish translation fixes and new strings (#68) 2022-02-24 13:46:21 +03:00
DYefremov
fd4325961c minor style changes 2022-02-23 14:22:29 +03:00
DYefremov
294d32c705 Russian, Belarusian and German translations update 2022-02-22 19:55:39 +03:00
DYefremov
aa961030ce changing path by link for FTP client 2022-02-22 14:44:55 +03:00
DYefremov
bc4c6746c9 minor corrections 2022-02-22 14:18:40 +03:00
DYefremov
61282b0cc8 copied tr *.mo file 2022-02-22 10:21:10 +03:00
audi06_19
07e855f99d Turkish translation update (#66) 2022-02-22 10:15:12 +03:00
DYefremov
0d0f19122b updated comments in *.desktop file 2022-02-21 20:33:24 +03:00
DYefremov
4789688efd updated es *.mo file 2022-02-21 20:09:40 +03:00
Víctor Pont
b13c2c3be0 Spanish translation update (#65) 2022-02-21 20:03:21 +03:00
DYefremov
d72189abc4 bump version 2022-02-21 15:03:17 +03:00
DYefremov
35d194100b added keyboard shortcut for renaming 2022-02-21 14:28:47 +03:00
DYefremov
aa0b97b9ae added extra tab for IPTV 2022-02-21 12:22:44 +03:00
DYefremov
5f54452ee2 fixed getting satellites list from FlySat 2022-02-21 00:23:07 +03:00
DYefremov
580e8ca82c basic bookmarks support for the FTP client 2022-02-13 00:29:44 +03:00
DYefremov
4ba2fb1a04 skip importing groups from m3u for Neutrino 2022-02-12 14:22:48 +03:00
DYefremov
b4612c26cb changed file names reading for FTP client 2022-02-09 21:56:45 +03:00
38 changed files with 2681 additions and 1120 deletions

View File

@@ -6,6 +6,7 @@ Comment[ru]=Редактор списка каналов и спутников
Comment[be]=Рэдактар спіса каналаў і спадарожнікаў для Enigma2
Comment[de]=Programm- und Satellitenlisten-Editor für Enigma2
Comment[tr]=Enigma2 için kanal ve uydu listesi editörü
Comment[es]=Editor de listas de canales y satélites para Enigma2
Icon=demon-editor
Exec=bash -c 'cd $(dirname %k) && ./start.py'
Terminal=false

View File

@@ -135,8 +135,8 @@ class UtfFTP(FTP):
files = []
self.dir(path, files.append)
for f in files:
f_data = f.split()
f_path = os.path.join(path, " ".join(f_data[8:]))
f_data = self.get_file_data(f)
f_path = f_data[8]
if f_data[0][0] == "d":
try:
@@ -310,9 +310,8 @@ class UtfFTP(FTP):
files = []
self.dir(path, files.append)
for f in files:
f_data = f.split()
name = " ".join(f_data[8:])
f_path = path + "/" + name
f_data = self.get_file_data(f)
f_path = f"{path}/{f_data[8]}"
if f_data[0][0] == "d":
self.delete_dir(f_path, callback)
@@ -351,6 +350,15 @@ class UtfFTP(FTP):
return resp
@staticmethod
def get_file_data(file):
""" Returns a prepared list of file data from a file string. """
f_data = file.split()
# Ignoring space in file name.
f_data = f_data[0:9]
f_data[8] = file[file.index(f_data[8]):]
return f_data
def download_data(*, settings, download_type=DownloadType.ALL, callback=log, files_filter=None):
with UtfFTP(host=settings.host, user=settings.user, passwd=settings.password) as ftp:

View File

@@ -2,7 +2,7 @@
#
# The MIT License (MIT)
#
# Copyright (c) 2018-2021 Dmitriy Yefremov
# Copyright (c) 2018-2022 Dmitriy Yefremov
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
@@ -40,6 +40,7 @@ from app.ui.uicommons import IPTV_ICON
NEUTRINO_FAV_ID_FORMAT = "{}::{}::{}::{}::{}::{}::{}::{}::{}::{}"
ENIGMA2_FAV_ID_FORMAT = " {}:{}:{}:{:X}:{:X}:{:X}:{:X}:0:0:0:{}:{}\n#DESCRIPTION: {}\n"
MARKER_FORMAT = " 1:64:{}:0:0:0:0:0:0:0::{}\n#DESCRIPTION {}\n"
PICON_FORMAT = "{}_{}_{:X}_{:X}_{:X}_{:X}_{:X}_0_0_0.png"
class StreamType(Enum):
@@ -49,6 +50,11 @@ class StreamType(Enum):
NONE_REC_2 = "5002"
E_SERVICE_URI = "8193"
E_SERVICE_HLS = "8739"
UNKNOWN = "0"
@classmethod
def _missing_(cls, value):
return cls.UNKNOWN
def parse_m3u(path, s_type, detect_encoding=True, params=None):
@@ -76,6 +82,7 @@ def parse_m3u(path, s_type, detect_encoding=True, params=None):
p_id = "1_0_1_0_0_0_0_0_0_0.png"
st = BqServiceType.IPTV.name
params = params or [0, 0, 0, 0]
m_name = BqServiceType.MARKER.name
for line in str(data, encoding=encoding, errors="ignore").splitlines():
if line.startswith("#EXTINF"):
@@ -88,26 +95,30 @@ def parse_m3u(path, s_type, detect_encoding=True, params=None):
d = {data[i].lower().strip(" ="): data[i + 1] for i in range(0, len(data) - 1, 2)}
picon = d.get("tvg-logo", None)
grp_name = d.get("group-title", None)
if grp_name not in groups:
groups.add(grp_name)
fav_id = MARKER_FORMAT.format(marker_counter, grp_name, grp_name)
marker_counter += 1
mr = Service(None, None, None, grp_name, *aggr[0:3], BqServiceType.MARKER.name, *aggr, fav_id, None)
services.append(mr)
if s_type is SettingsType.ENIGMA_2:
grp_name = d.get("group-title", None)
if grp_name not in groups:
groups.add(grp_name)
fav_id = MARKER_FORMAT.format(marker_counter, grp_name, grp_name)
marker_counter += 1
mr = Service(None, None, None, grp_name, *aggr[0:3], m_name, *aggr, fav_id, None)
services.append(mr)
elif line.startswith("#EXTGRP") and s_type is SettingsType.ENIGMA_2:
grp_name = line.strip("#EXTGRP:").strip()
if grp_name not in groups:
groups.add(grp_name)
fav_id = MARKER_FORMAT.format(marker_counter, grp_name, grp_name)
marker_counter += 1
mr = Service(None, None, None, grp_name, *aggr[0:3], BqServiceType.MARKER.name, *aggr, fav_id, None)
mr = Service(None, None, None, grp_name, *aggr[0:3], m_name, *aggr, fav_id, None)
services.append(mr)
elif not line.startswith("#"):
url = line.strip()
params[0] = sid_counter
sid_counter += 1
fav_id = get_fav_id(url, name, s_type, params)
if s_type is SettingsType.ENIGMA_2:
p_id = get_picon_id(params)
if all((name, url, fav_id)):
srv = Service(None, None, IPTV_ICON, name, *aggr[0:3], st, picon, p_id, *s_aggr, url, fav_id, None)
services.append(srv)
@@ -152,5 +163,11 @@ def get_fav_id(url, name, settings_type, params=None, st_type=None, s_id=0, srv_
return NEUTRINO_FAV_ID_FORMAT.format(url, "", 0, None, None, None, None, "", "", 1)
def get_picon_id(params=None, st_type=None, s_id=0, srv_type=1):
st_type = st_type or StreamType.NONE_TS.value
params = params or (0, 0, 0, 0)
return PICON_FORMAT.format(st_type, s_id, srv_type, *params)
if __name__ == "__main__":
pass

View File

@@ -503,6 +503,16 @@ class Settings:
def epg_options(self, value):
self._cp_settings["epg_options"] = value
# *********** FTP ************ #
@property
def ftp_bookmarks(self):
return self._cp_settings.get("ftp_bookmarks", [])
@ftp_bookmarks.setter
def ftp_bookmarks(self, value):
self._cp_settings["ftp_bookmarks"] = value
# ***** Program settings ***** #
@property

View File

@@ -57,7 +57,7 @@ class PiconsCzDownloader:
_PERM_URL = "https://picon.cz/download/7337"
_BASE_URL = "https://picon.cz/download/"
_BASE_LOGO_URL = "https://picon.cz/picon/0/"
_HEADER = {"User-Agent": "DemonEditor/2.1.2", "Referer": ""}
_HEADER = {"User-Agent": "DemonEditor/2.2.0", "Referer": ""}
_LINK_PATTERN = re.compile(r"((.*)-\d+x\d+)-(.*)_by_chocholousek.7z$")
_FILE_PATTERN = re.compile(b"\\s+(1_.*\\.png).*")

View File

@@ -195,36 +195,37 @@ class SatellitesParser(HTMLParser):
names = []
pos = ""
pos_url = ""
satellites = []
def normalize_pos(p):
return f"{float(p[:-1])}{p[-1]}" if "." not in p else p
def get_sat(r):
nonlocal pos
nonlocal pos_url
# Uniting satellites in position.
if re.match(pos_pat, r[2]):
pos_url = r[2]
name = r[1]
pos = normalize_pos(self.parse_position(r[3]))
for row in filter(lambda x: len(x) > 6, self._rows):
if re.match(sat_pat, row[1]):
row.pop(0)
names.append(name)
return name, pos, r[5], r[0], False
r_size = len(r)
if r_size == 5:
name = r[1]
names.append(name)
return name, pos, r[3], r[0], False
if r_size == 6:
if names:
name = "/".join(names)
if re.match(sat_pat, row[0]) and row[-2]: # r[-2] -> skip EMPTY satellites!
if re.match(pos_pat, row[0]):
names.clear()
return name, pos, None, pos_url, False
pos_url = row[0]
name = row[3]
pos = normalize_pos(self.parse_position(row[-4]))
names.append(name)
satellites.append((name, pos, row[-2], row[2], False))
return r[1], normalize_pos(self.parse_position(r[2])), r[4], r[0], False
if len(row) == 7:
single_pos = normalize_pos(self.parse_position(row[-4]))
name = row[1]
if pos == single_pos:
names.append(name)
else:
# Uniting satellites in position.
if len(names) > 1:
satellites.append(("/".join(names), pos, None, pos_url, False))
names.clear()
satellites.append((name, single_pos, row[-2], row[0], False))
return list(filter(None, map(get_sat, filter(lambda row: row and re.match(sat_pat, row[0]), self._rows))))
return satellites
def get_satellites_for_lyng_sat(self):
base_url = "https://www.lyngsat.com/"

View File

@@ -617,10 +617,10 @@ class TimerTool(Gtk.Box):
return
fav_id = None
if source == self._app.FAV_MODEL_NAME:
if source == self._app.FAV_MODEL:
model = self._app.fav_view.get_model()
fav_id = model.get_value(model.get_iter_from_string(itrs[0]), Column.FAV_ID)
elif source == self._app.SERVICE_MODEL_NAME:
elif source == self._app.SERVICE_MODEL:
model = self._app.services_view.get_model()
fav_id = model.get_value(model.get_iter_from_string(itrs[0]), Column.SRV_FAV_ID)
@@ -727,7 +727,7 @@ class RecordingsTool(Gtk.Box):
model.append((None, self.ROOT, self._ftp.pwd()))
for f in files:
f_data = f.split()
f_data = self._ftp.get_file_data(f)
if len(f_data) < 9:
log(f"{__class__.__name__}. Folder data parsing error. [{f}]")
continue
@@ -735,7 +735,7 @@ class RecordingsTool(Gtk.Box):
f_type = f_data[0][0]
if f_type == "d":
model.append((self._icon, " ".join(f_data[8:]), self._ftp.pwd()))
model.append((self._icon, f_data[8], self._ftp.pwd()))
def on_path_activated(self, view, path, column):
row = view.get_model()[path][:]

View File

@@ -31,7 +31,7 @@ Author: Dmitriy Yefremov
<!-- interface-license-type mit -->
<!-- interface-name DemonEditor -->
<!-- interface-description Enigma2 channel and satellites list editor. -->
<!-- interface-copyright 2018-2021 Dmitriy Yefremov -->
<!-- interface-copyright 2018-2022 Dmitriy Yefremov -->
<!-- interface-authors Dmitriy Yefremov -->
<object class="GtkAboutDialog" id="about_dialog">
<property name="can_focus">False</property>
@@ -40,8 +40,8 @@ Author: Dmitriy Yefremov
<property name="icon_name">system-help</property>
<property name="type_hint">normal</property>
<property name="program_name">DemonEditor</property>
<property name="version">2.1.2 Beta</property>
<property name="copyright">2018-2021 Dmitriy Yefremov
<property name="version">2.2.0 Beta</property>
<property name="copyright">2018-2022 Dmitriy Yefremov
</property>
<property name="comments" translatable="yes">Enigma2 channel and satellite list editor.</property>
<property name="website">https://dyefremov.github.io/DemonEditor/</property>

View File

@@ -34,11 +34,23 @@ Author: Dmitriy Yefremov
<!-- interface-description Enigma2 channel and satellite list editor for GNU/Linux. -->
<!-- interface-copyright 2018-2020 Dmitriy Yefremov -->
<!-- interface-authors Dmitriy Yefremov -->
<object class="GtkMenu" id="bookmark_popup_menu">
<property name="visible">True</property>
<property name="can_focus">False</property>
<child>
<object class="GtkImageMenuItem" id="bookmark_remove_menu_item">
<property name="label">gtk-remove</property>
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="use_underline">True</property>
<property name="use_stock">True</property>
<signal name="activate" handler="on_bookmark_remove" swapped="no"/>
</object>
</child>
</object>
<object class="GtkListStore" id="bookmarks_list_store">
<columns>
<!-- column-name name -->
<column type="gchararray"/>
<!-- column-name url -->
<!-- column-name path -->
<column type="gchararray"/>
</columns>
</object>
@@ -174,9 +186,66 @@ Author: Dmitriy Yefremov
<property name="position">1</property>
</packing>
</child>
<child>
<object class="GtkButtonBox" id="bookmark_button_box">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="homogeneous">True</property>
<property name="layout_style">expand</property>
<child>
<object class="GtkToggleButton" id="bookmarks_button">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="receives_default">True</property>
<property name="tooltip_text" translatable="yes">Bookmarks</property>
<child>
<object class="GtkImage" id="bookmark_image">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="icon_name">user-bookmarks-symbolic</property>
</object>
</child>
</object>
<packing>
<property name="expand">True</property>
<property name="fill">True</property>
<property name="position">1</property>
</packing>
</child>
<child>
<object class="GtkButton" id="add_ftp_bookmark_button">
<property name="visible">True</property>
<property name="sensitive">False</property>
<property name="can_focus">False</property>
<property name="receives_default">True</property>
<property name="tooltip_text" translatable="yes">Add bookmark</property>
<signal name="clicked" handler="on_bookmark_add" swapped="no"/>
<child>
<object class="GtkImage" id="ftp_remove_button_image1">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="icon_name">bookmark-new-symbolic</property>
</object>
</child>
</object>
<packing>
<property name="expand">False</property>
<property name="fill">True</property>
<property name="pack_type">end</property>
<property name="position">1</property>
</packing>
</child>
</object>
<packing>
<property name="expand">False</property>
<property name="fill">True</property>
<property name="position">2</property>
</packing>
</child>
<child>
<object class="GtkBox" id="ftp_actions_box">
<property name="visible">True</property>
<property name="sensitive">False</property>
<property name="can_focus">False</property>
<property name="spacing">5</property>
<child>
@@ -184,7 +253,7 @@ Author: Dmitriy Yefremov
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="receives_default">True</property>
<property name="tooltip_text" translatable="yes">Add folder</property>
<property name="tooltip_text" translatable="yes">New folder</property>
<signal name="clicked" handler="on_ftp_create_folder" object="ftp_name_column_renderer" swapped="no"/>
<child>
<object class="GtkImage" id="ftp_add_folder_button_image">
@@ -246,20 +315,7 @@ Author: Dmitriy Yefremov
<packing>
<property name="expand">False</property>
<property name="fill">True</property>
<property name="position">2</property>
</packing>
</child>
<child>
<object class="GtkComboBox" id="bookmark_button">
<property name="can_focus">False</property>
<property name="model">bookmarks_list_store</property>
<property name="id_column">0</property>
</object>
<packing>
<property name="expand">False</property>
<property name="fill">True</property>
<property name="pack_type">end</property>
<property name="position">5</property>
<property name="position">3</property>
</packing>
</child>
</object>
@@ -270,122 +326,191 @@ Author: Dmitriy Yefremov
</packing>
</child>
<child>
<object class="GtkScrolledWindow" id="ftp_view_scrolled_window">
<object class="GtkBox" id="ftp_data_box">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="shadow_type">in</property>
<property name="min_content_height">100</property>
<property name="can_focus">False</property>
<property name="spacing">5</property>
<child>
<object class="GtkTreeView" id="ftp_view">
<object class="GtkBox" id="bookmarks_box">
<property name="width_request">150</property>
<property name="can_focus">False</property>
<property name="orientation">vertical</property>
<property name="spacing">2</property>
<child>
<object class="GtkScrolledWindow" id="bookmarks_view_scrolled_window">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="shadow_type">in</property>
<child>
<object class="GtkTreeView" id="bookmarks_view">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="model">bookmarks_list_store</property>
<property name="headers_visible">False</property>
<property name="search_column">0</property>
<property name="tooltip_column">0</property>
<signal name="button-press-event" handler="on_view_popup_menu" object="bookmark_popup_menu" swapped="no"/>
<signal name="key-press-event" handler="on_view_key_press" swapped="no"/>
<signal name="row-activated" handler="on_bookmark_activated" swapped="no"/>
<child internal-child="selection">
<object class="GtkTreeSelection" id="bookmarks_selection"/>
</child>
<child>
<object class="GtkTreeViewColumn" id="bookmark_name_column">
<property name="title" translatable="yes">Name</property>
<child>
<object class="GtkCellRendererText" id="bookmark_name_renderer">
<property name="ellipsize">end</property>
</object>
<attributes>
<attribute name="text">0</attribute>
</attributes>
</child>
</object>
</child>
</object>
</child>
</object>
<packing>
<property name="expand">True</property>
<property name="fill">True</property>
<property name="position">0</property>
</packing>
</child>
<child>
<placeholder/>
</child>
</object>
<packing>
<property name="expand">False</property>
<property name="fill">True</property>
<property name="position">0</property>
</packing>
</child>
<child>
<object class="GtkScrolledWindow" id="ftp_view_scrolled_window">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="model">ftp_list_store</property>
<property name="search_column">1</property>
<property name="rubber_banding">True</property>
<signal name="button-press-event" handler="on_view_popup_menu" object="ftp_popup_menu" swapped="no"/>
<signal name="button-press-event" handler="on_view_press" swapped="no"/>
<signal name="button-release-event" handler="on_view_release" swapped="no"/>
<signal name="drag-begin" handler="on_view_drag_begin" after="yes" swapped="no"/>
<signal name="drag-data-get" handler="on_ftp_drag_data_get" swapped="no"/>
<signal name="drag-data-received" handler="on_ftp_drag_data_received" swapped="no"/>
<signal name="drag-end" handler="on_view_drag_end" swapped="no"/>
<signal name="key-press-event" handler="on_view_key_press" swapped="no"/>
<signal name="row-activated" handler="on_ftp_row_activated" swapped="no"/>
<child internal-child="selection">
<object class="GtkTreeSelection" id="ftp_selection">
<property name="mode">multiple</property>
</object>
</child>
<property name="shadow_type">in</property>
<property name="min_content_height">100</property>
<child>
<object class="GtkTreeViewColumn" id="ftp_name_column">
<property name="resizable">True</property>
<property name="min_width">100</property>
<property name="title" translatable="yes">Name</property>
<property name="expand">True</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">1</property>
<child>
<object class="GtkCellRendererPixbuf" id="ftp_icon_column_renderer">
<property name="xalign">0.019999999552965164</property>
<object class="GtkTreeView" id="ftp_view">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="model">ftp_list_store</property>
<property name="search_column">1</property>
<property name="rubber_banding">True</property>
<signal name="button-press-event" handler="on_view_popup_menu" object="ftp_popup_menu" swapped="no"/>
<signal name="button-press-event" handler="on_view_press" swapped="no"/>
<signal name="button-release-event" handler="on_view_release" swapped="no"/>
<signal name="drag-begin" handler="on_view_drag_begin" after="yes" swapped="no"/>
<signal name="drag-data-get" handler="on_ftp_drag_data_get" swapped="no"/>
<signal name="drag-data-received" handler="on_ftp_drag_data_received" swapped="no"/>
<signal name="drag-end" handler="on_view_drag_end" swapped="no"/>
<signal name="key-press-event" handler="on_view_key_press" swapped="no"/>
<signal name="row-activated" handler="on_ftp_row_activated" swapped="no"/>
<child internal-child="selection">
<object class="GtkTreeSelection" id="ftp_selection">
<property name="mode">multiple</property>
</object>
<attributes>
<attribute name="pixbuf">0</attribute>
</attributes>
</child>
<child>
<object class="GtkCellRendererText" id="ftp_name_column_renderer">
<property name="xalign">0.019999999552965164</property>
<property name="ellipsize">end</property>
<signal name="edited" handler="on_ftp_renamed" swapped="no"/>
<object class="GtkTreeViewColumn" id="ftp_name_column">
<property name="resizable">True</property>
<property name="min_width">100</property>
<property name="title" translatable="yes">Name</property>
<property name="expand">True</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">1</property>
<child>
<object class="GtkCellRendererPixbuf" id="ftp_icon_column_renderer">
<property name="xalign">0.019999999552965164</property>
</object>
<attributes>
<attribute name="pixbuf">0</attribute>
</attributes>
</child>
<child>
<object class="GtkCellRendererText" id="ftp_name_column_renderer">
<property name="xalign">0.019999999552965164</property>
<property name="ellipsize">end</property>
<signal name="edited" handler="on_ftp_renamed" swapped="no"/>
</object>
<attributes>
<attribute name="text">1</attribute>
</attributes>
</child>
</object>
<attributes>
<attribute name="text">1</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="ftp_size_column">
<property name="sizing">fixed</property>
<property name="min_width">75</property>
<property name="title" translatable="yes">Size</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">2</property>
<child>
<object class="GtkCellRendererText" id="ftp_size_column_renderer">
<property name="xalign">0.94999998807907104</property>
<object class="GtkTreeViewColumn" id="ftp_size_column">
<property name="sizing">fixed</property>
<property name="min_width">75</property>
<property name="title" translatable="yes">Size</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">2</property>
<child>
<object class="GtkCellRendererText" id="ftp_size_column_renderer">
<property name="xalign">0.94999998807907104</property>
</object>
<attributes>
<attribute name="text">2</attribute>
</attributes>
</child>
</object>
<attributes>
<attribute name="text">2</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="ftp_date_column">
<property name="min_width">75</property>
<property name="title" translatable="yes">Date</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">3</property>
<child>
<object class="GtkCellRendererText" id="ftp_date_column_renderer"/>
<attributes>
<attribute name="text">3</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="ftp_attr_column">
<property name="sizing">fixed</property>
<property name="min_width">85</property>
<property name="title" translatable="yes">Attr.</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">4</property>
<child>
<object class="GtkCellRendererText" id="ftp_attr_column_renderer">
<property name="xalign">0.50999999046325684</property>
<property name="ellipsize">end</property>
<object class="GtkTreeViewColumn" id="ftp_date_column">
<property name="min_width">75</property>
<property name="title" translatable="yes">Date</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">3</property>
<child>
<object class="GtkCellRendererText" id="ftp_date_column_renderer"/>
<attributes>
<attribute name="text">3</attribute>
</attributes>
</child>
</object>
<attributes>
<attribute name="text">4</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="ftp_extra_column">
<property name="visible">False</property>
<property name="title" translatable="yes">Extra</property>
<child>
<object class="GtkCellRendererText" id="ftp_extra_column_renderer"/>
<attributes>
<attribute name="text">5</attribute>
</attributes>
<object class="GtkTreeViewColumn" id="ftp_attr_column">
<property name="sizing">fixed</property>
<property name="min_width">85</property>
<property name="title" translatable="yes">Attr.</property>
<property name="alignment">0.5</property>
<property name="sort_column_id">4</property>
<child>
<object class="GtkCellRendererText" id="ftp_attr_column_renderer">
<property name="xalign">0.50999999046325684</property>
<property name="ellipsize">end</property>
</object>
<attributes>
<attribute name="text">4</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="ftp_extra_column">
<property name="visible">False</property>
<property name="title" translatable="yes">Extra</property>
<child>
<object class="GtkCellRendererText" id="ftp_extra_column_renderer"/>
<attributes>
<attribute name="text">5</attribute>
</attributes>
</child>
</object>
</child>
</object>
</child>
</object>
<packing>
<property name="expand">True</property>
<property name="fill">True</property>
<property name="position">1</property>
</packing>
</child>
</object>
<packing>
@@ -395,7 +520,7 @@ Author: Dmitriy Yefremov
</packing>
</child>
<child>
<object class="GtkBox">
<object class="GtkBox" id="ftp_status_bar_box">
<property name="height_request">24</property>
<property name="visible">True</property>
<property name="can_focus">False</property>
@@ -484,7 +609,7 @@ Author: Dmitriy Yefremov
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="receives_default">True</property>
<property name="tooltip_text" translatable="yes">Add folder</property>
<property name="tooltip_text" translatable="yes">New folder</property>
<signal name="clicked" handler="on_file_create_folder" object="file_name_column_renderer" swapped="no"/>
<child>
<object class="GtkImage" id="pc_add_folder_button_image">
@@ -649,7 +774,7 @@ Author: Dmitriy Yefremov
</packing>
</child>
<child>
<object class="GtkBox">
<object class="GtkBox" id="file_status_bar_box">
<property name="height_request">24</property>
<property name="visible">True</property>
<property name="can_focus">False</property>
@@ -747,6 +872,7 @@ Author: Dmitriy Yefremov
<object class="GtkImageMenuItem" id="ftp_create_folder_menu_item">
<property name="label" translatable="yes">Create folder</property>
<property name="visible">True</property>
<property name="sensitive">False</property>
<property name="can_focus">False</property>
<property name="image">ftp_create_folder_image</property>
<property name="use_stock">False</property>
@@ -758,6 +884,7 @@ Author: Dmitriy Yefremov
<object class="GtkImageMenuItem" id="ftp_edit_menu_item">
<property name="label">gtk-edit</property>
<property name="visible">True</property>
<property name="sensitive">False</property>
<property name="can_focus">False</property>
<property name="use_underline">True</property>
<property name="use_stock">True</property>
@@ -770,6 +897,7 @@ Author: Dmitriy Yefremov
<property name="label" translatable="yes">Rename</property>
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="sensitive">False</property>
<property name="image">rename_image</property>
<property name="use_stock">False</property>
<signal name="activate" handler="on_ftp_rename" object="ftp_name_column_renderer" swapped="no"/>
@@ -787,6 +915,7 @@ Author: Dmitriy Yefremov
<object class="GtkImageMenuItem" id="ftp_remove_menu_item">
<property name="label" translatable="yes">Remove</property>
<property name="visible">True</property>
<property name="sensitive">False</property>
<property name="can_focus">False</property>
<property name="image">remove_image_2</property>
<property name="use_stock">False</property>

View File

@@ -159,6 +159,7 @@ class FtpClientBox(Gtk.HBox):
"on_disconnect": self.on_disconnect,
"on_ftp_row_activated": self.on_ftp_row_activated,
"on_file_row_activated": self.on_file_row_activated,
"on_bookmark_activated": self.on_bookmark_activated,
"on_ftp_edit": self.on_ftp_edit,
"on_ftp_rename": self.on_ftp_rename,
"on_ftp_renamed": self.on_ftp_renamed,
@@ -168,6 +169,7 @@ class FtpClientBox(Gtk.HBox):
"on_file_copy": self.on_file_copy,
"on_file_remove": self.on_file_remove,
"on_ftp_remove": self.on_ftp_file_remove,
"on_bookmark_remove": self.on_bookmark_remove,
"on_file_create_folder": self.on_file_create_folder,
"on_ftp_create_folder": self.on_ftp_create_folder,
"on_view_drag_begin": self.on_view_drag_begin,
@@ -176,6 +178,7 @@ class FtpClientBox(Gtk.HBox):
"on_file_drag_data_get": self.on_file_drag_data_get,
"on_file_drag_data_received": self.on_file_drag_data_received,
"on_view_drag_end": self.on_view_drag_end,
"on_bookmark_add": self.on_bookmark_add,
"on_view_popup_menu": on_popup_menu,
"on_view_key_press": self.on_view_key_press,
"on_view_press": self.on_view_press,
@@ -192,6 +195,8 @@ class FtpClientBox(Gtk.HBox):
self._file_view = builder.get_object("file_view")
self._file_model = builder.get_object("file_list_store")
self._file_name_renderer = builder.get_object("file_name_column_renderer")
self._bookmark_view = builder.get_object("bookmarks_view")
self._bookmark_model = builder.get_object("bookmarks_list_store")
# Buttons
self._connect_button = builder.get_object("connect_button")
disconnect_button = builder.get_object("disconnect_button")
@@ -200,7 +205,10 @@ class FtpClientBox(Gtk.HBox):
disconnect_button.bind_property("visible", builder.get_object("ftp_edit_menu_item"), "sensitive")
disconnect_button.bind_property("visible", builder.get_object("ftp_rename_menu_item"), "sensitive")
disconnect_button.bind_property("visible", builder.get_object("ftp_remove_menu_item"), "sensitive")
disconnect_button.bind_property("visible", builder.get_object("add_ftp_bookmark_button"), "sensitive")
self._connect_button.bind_property("visible", builder.get_object("disconnect_button"), "visible", 4)
self._bookmarks_button = builder.get_object("bookmarks_button")
self._bookmarks_button.bind_property("active", builder.get_object("bookmarks_box"), "visible")
# Force Ctrl
self._ftp_view.connect("key-press-event", self._app.force_ctrl)
self._file_view.connect("key-press-event", self._app.force_ctrl)
@@ -218,6 +226,7 @@ class FtpClientBox(Gtk.HBox):
@run_task
def init_ftp(self):
self.init_bookmarks()
GLib.idle_add(self._ftp_model.clear)
try:
if self._ftp:
@@ -291,7 +300,7 @@ class FtpClientBox(Gtk.HBox):
self._ftp_model.append(File(None, self.ROOT, None, None, self._ftp.pwd(), "0"))
for f in files:
f_data = f.split()
f_data = self._ftp.get_file_data(f)
f_type = f_data[0][0]
is_dir = f_type == "d"
is_link = f_type == "l"
@@ -308,7 +317,7 @@ class FtpClientBox(Gtk.HBox):
r_size = self.get_size_from_bytes(size)
date = f"{f_data[5]}, {f_data[6]} {f_data[7]}"
self._ftp_model.append(File(icon, " ".join(f_data[8:]), r_size, date, f_data[0], size))
self._ftp_model.append(File(icon, f_data[8], r_size, date, f_data[0], size))
def on_connect(self, item=None):
self.init_ftp()
@@ -326,6 +335,10 @@ class FtpClientBox(Gtk.HBox):
if size == self.FOLDER or f_path == self.ROOT:
self.init_ftp_data(f_path)
elif size == self.LINK:
name, sep, f_path = f_path.partition("->")
if f_path:
self.init_ftp_data(f_path.strip())
else:
b_size = row[self.Column.EXTRA]
if b_size.isdigit() and int(b_size) > self.MAX_SIZE:
@@ -712,6 +725,27 @@ class FtpClientBox(Gtk.HBox):
self._ftp_info_label.set_text(message)
self._ftp_info_label.set_tooltip_text(message)
# **************** Bookmarks ***************** #
@run_idle
def init_bookmarks(self):
self._bookmark_model.clear()
list(map(lambda b: self._bookmark_model.append((b,)), self._settings.ftp_bookmarks))
def on_bookmark_activated(self, view, path, column):
self.init_ftp_data(self._bookmark_model[path][0])
def on_bookmark_add(self, item=None):
self._bookmarks_button.set_active(True)
self._bookmark_model.append((self._ftp_model.get_value(self._ftp_model.get_iter_first(), 4),))
self._settings.ftp_bookmarks = [r[0] for r in self._bookmark_model]
def on_bookmark_remove(self, item=None):
model, paths = self._bookmark_view.get_selection().get_selected_rows()
if paths and show_dialog(DialogType.QUESTION, self._app.app_window) == Gtk.ResponseType.OK:
list(map(lambda p: model.remove(model.get_iter(p)), paths))
self._settings.ftp_bookmarks = [r[0] for r in self._bookmark_model]
def on_view_key_press(self, view, event):
key_code = event.hardware_keycode
if not KeyboardKey.value_exist(key_code):
@@ -743,6 +777,8 @@ class FtpClientBox(Gtk.HBox):
self.on_ftp_file_remove()
elif self._file_view.is_focus():
self.on_file_remove()
elif self._bookmark_view.is_focus():
self.on_bookmark_remove()
elif key is KeyboardKey.RETURN:
path, column = view.get_cursor()
if path:
@@ -760,11 +796,13 @@ class FtpClientBox(Gtk.HBox):
# Enable selection.
self._select_enabled = True
def on_paned_size_allocate(self, paned, allocation):
@staticmethod
def on_paned_size_allocate(paned, allocation):
""" Sets default homogeneous sizes. """
paned.set_position(0.5 * allocation.width)
def get_size_from_bytes(self, size):
@staticmethod
def get_size_from_bytes(size):
""" Simple convert function from bytes to other units like K, M or G. """
try:
b = float(size)

View File

@@ -2,7 +2,7 @@
#
# The MIT License (MIT)
#
# Copyright (c) 2018-2021 Dmitriy Yefremov
# Copyright (c) 2018-2022 Dmitriy Yefremov
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
@@ -44,7 +44,7 @@ from app.eparser.iptv import (NEUTRINO_FAV_ID_FORMAT, StreamType, ENIGMA2_FAV_ID
from app.settings import SettingsType
from app.tools.yt import YouTubeException, YouTube
from app.ui.dialogs import Action, show_dialog, DialogType, get_message, get_builder
from app.ui.main_helper import get_base_model, get_iptv_url, on_popup_menu, get_picon_pixbuf
from app.ui.main_helper import get_iptv_url, on_popup_menu, get_picon_pixbuf
from app.ui.uicommons import (Gtk, Gdk, UI_RESOURCES_PATH, IPTV_ICON, Column, KeyboardKey, get_yt_icon)
_DIGIT_ENTRY_NAME = "digit-entry"
@@ -77,7 +77,7 @@ def get_stream_type(box):
class IptvDialog:
def __init__(self, transient, view, services, bouquet, settings, action=Action.ADD):
def __init__(self, app, view, bouquet=None, service=None, action=Action.ADD):
handlers = {"on_response": self.on_response,
"on_entry_changed": self.on_entry_changed,
"on_url_changed": self.on_url_changed,
@@ -86,11 +86,11 @@ class IptvDialog:
"on_yt_quality_changed": self.on_yt_quality_changed,
"on_info_bar_close": self.on_info_bar_close}
self._app = app
self._action = action
self._s_type = settings.setting_type
self._settings = settings
self._settings = app.app_settings
self._s_type = self._settings.setting_type
self._bouquet = bouquet
self._services = services
self._yt_links = None
self._yt_dl = None
@@ -98,7 +98,7 @@ class IptvDialog:
objects=("iptv_dialog", "stream_type_liststore", "yt_quality_liststore"))
self._dialog = builder.get_object("iptv_dialog")
self._dialog.set_transient_for(transient)
self._dialog.set_transient_for(app.app_window)
self._name_entry = builder.get_object("name_entry")
self._description_entry = builder.get_object("description_entry")
self._url_entry = builder.get_object("url_entry")
@@ -142,7 +142,7 @@ class IptvDialog:
self.update_reference_entry()
self._stream_type_combobox.set_active(1)
elif self._action is Action.EDIT:
self._current_srv = get_base_model(self._model)[self._paths][:]
self._current_srv = service
self.init_data(self._current_srv)
def show(self):
@@ -167,8 +167,8 @@ class IptvDialog:
self._dialog.destroy()
def init_data(self, srv):
name, fav_id = srv[2], srv[7]
self._name_entry.set_text(name)
fav_id = srv.fav_id
self._name_entry.set_text(srv.service)
self.init_enigma2_data(fav_id) if self._s_type is SettingsType.ENIGMA_2 else self.init_neutrino_data(fav_id)
def init_enigma2_data(self, fav_id):
@@ -307,11 +307,12 @@ class IptvDialog:
int(self._namespace_entry.get_text()),
quote(self._url_entry.get_text()),
name, name)
self.update_bouquet_data(name, fav_id)
def save_neutrino_data(self):
if self._action is Action.EDIT:
id_data = self._current_srv[7].split("::")
id_data = self._current_srv.fav_id.split("::")
else:
id_data = ["", "", "0", None, None, None, None, "", "", "1"]
id_data[0] = self._url_entry.get_text()
@@ -320,20 +321,25 @@ class IptvDialog:
self._dialog.destroy()
def update_bouquet_data(self, name, fav_id):
picon_id = f"{self._reference_entry.get_text().replace(':', '_')}.png"
if self._action is Action.EDIT:
old_srv = self._services.pop(self._current_srv[7])
self._services[fav_id] = old_srv._replace(service=name, fav_id=fav_id)
self._bouquet[self._paths[0][0]] = fav_id
self._model.set(self._model.get_iter(self._paths), {Column.FAV_SERVICE: name, Column.FAV_ID: fav_id})
services = self._app.current_services
old_srv = services.pop(self._current_srv.fav_id)
new_service = old_srv._replace(service=name, fav_id=fav_id, picon_id=picon_id)
services[fav_id] = new_service
self._app.emit("iptv-service-edited", (old_srv, new_service))
else:
aggr = [None] * 10
aggr = [None] * 8
s_type = BqServiceType.IPTV.name
srv = (None, None, name, None, None, s_type, None, fav_id, *aggr[0:3])
itr = self._model.insert_after(self._model.get_iter(self._paths[0]),
srv) if self._paths else self._model.insert(0, srv)
self._model.set_value(itr, 1, IPTV_ICON)
self._bouquet.insert(self._model.get_path(itr)[0], fav_id)
self._services[fav_id] = Service(None, None, IPTV_ICON, name, *aggr[0:3], s_type, *aggr, fav_id, None)
service = Service(None, None, IPTV_ICON, name, *aggr[0:3], s_type, None, picon_id, *aggr, fav_id, None)
self._app.current_services[fav_id] = service
self._app.emit("iptv-service-added", (service,))
@run_idle
def on_info_bar_close(self, bar=None, resp=None):
@@ -833,7 +839,7 @@ class M3uImportDialog(IptvListDialog):
class YtListImportDialog:
def __init__(self, transient, settings, appender):
def __init__(self, app):
handlers = {"on_import": self.on_import,
"on_receive": self.on_receive,
"on_yt_url_entry_changed": self.on_url_entry_changed,
@@ -845,12 +851,13 @@ class YtListImportDialog:
"on_key_press": self.on_key_press,
"on_close": self.on_close}
self.appender = appender
self._s_type = settings.setting_type
# self._main_window, self._settings, self.append_imported_services
self.appender = app.append_imported_services
self._settings = app.app_settings
self._s_type = self._settings.setting_type
self._download_task = False
self._yt_list_id = None
self._yt_list_title = None
self._settings = settings
self._yt = None
builder = get_builder(_UI_PATH, handlers, use_str=True,
@@ -859,7 +866,7 @@ class YtListImportDialog:
"yt_import_image"))
self._dialog = builder.get_object("yt_import_dialog_window")
self._dialog.set_transient_for(transient)
self._dialog.set_transient_for(app.app_window)
self._list_view_scrolled_window = builder.get_object("yt_list_view_scrolled_window")
self._model = builder.get_object("yt_liststore")
self._progress_bar = builder.get_object("yt_progress_bar")

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@@ -360,7 +360,7 @@ def has_locked_hide(model, paths, col_num):
# ***************** Location *******************#
def locate_in_services(fav_view, services_view, parent_window):
def locate_in_services(fav_view, services_view, column, parent_window):
""" Locating and scrolling to the service """
model, paths = fav_view.get_selection().get_selected_rows()
@@ -372,7 +372,7 @@ def locate_in_services(fav_view, services_view, parent_window):
fav_id = model.get_value(model.get_iter(paths[0]), Column.FAV_ID)
for index, row in enumerate(services_view.get_model()):
if row[Column.SRV_FAV_ID] == fav_id:
if row[column] == fav_id:
scroll_to(index, services_view)
break

View File

@@ -377,7 +377,7 @@ class PiconManager(Gtk.Box):
return
itr_str, sep, src = txt.partition(self._app.DRAG_SEP)
if src == self._app.BQ_MODEL_NAME:
if src == self._app.BQ_MODEL:
return
path, pos = view.get_dest_row_at_pos(x, y) or (None, None)
@@ -385,7 +385,7 @@ class PiconManager(Gtk.Box):
return
model = view.get_model()
if src == self._app.FAV_MODEL_NAME:
if src == self._app.FAV_MODEL:
target_view = self._app.fav_view
c_id = Column.FAV_ID
else:

View File

@@ -2,7 +2,7 @@
#
# The MIT License (MIT)
#
# Copyright (c) 2018-2021 Dmitriy Yefremov
# Copyright (c) 2018-2022 Dmitriy Yefremov
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
@@ -62,7 +62,7 @@ class ServiceDetailsDialog:
_DIGIT_ENTRY_NAME = "digit-entry"
def __init__(self, transient, settings, srv_view, fav_view, services, bouquets, new_color, action=Action.EDIT):
def __init__(self, app, new_color, action=Action.EDIT):
handlers = {"on_system_changed": self.on_system_changed,
"on_save": self.on_save,
"on_create_new": self.on_create_new,
@@ -76,19 +76,19 @@ class ServiceDetailsDialog:
builder = get_builder(_UI_PATH, handlers, use_str=True)
self._builder = builder
settings = app.app_settings
self._dialog = builder.get_object("service_details_dialog")
self._dialog.set_transient_for(transient)
self._dialog.set_transient_for(app.app_window)
self._s_type = settings.setting_type
self._tr_type = TrType.Satellite
self._satellites_xml_path = settings.profile_data_path + "satellites.xml"
self._picons_path = settings.profile_picons_path
self._services_view = srv_view
self._fav_view = fav_view
self._services_view = app.services_view
self._fav_view = app.fav_view
self._action = action
self._old_service = None
self._services = services
self._bouquets = bouquets
self._services = app.current_services
self._bouquets = app.current_bouquets
self._new_color = new_color
self._transponder_services_iters = None
self._current_model = None

View File

@@ -1,3 +1,7 @@
* {
-GtkDialog-action-area-border: 6;
}
#digit-entry {
border-color: Red;
}
@@ -10,12 +14,24 @@
margin: 1px;
}
#stack-switch-button {
padding-top: 2px;
padding-bottom: 2px;
}
paned > separator {
background-repeat: no-repeat;
background-position: center;
}
paned.horizontal > separator {
background-size: 2px 24px;
}
paned.vertical > separator {
background-size: 24px 2px;
}
.red-button {
background-image: none;
background-color: red;

View File

@@ -189,6 +189,7 @@ class ViewTarget(Enum):
BOUQUET = 0
FAV = 1
SERVICES = 2
IPTV = 3
class BqGenType(Enum):
@@ -259,6 +260,13 @@ class Column(IntEnum):
REC_LEN = 3
REC_FILE = 4
REC_DESC = 5
# IPTV view
IPTV_SERVICE = 0
IPTV_TYPE = 1
IPTV_PICON = 2
IPTV_REF = 3
IPTV_FAV_ID = 4
IPTV_PICON_ID = 5
def __index__(self):
""" Overridden to get the index in slices directly """

View File

@@ -1,16 +1,11 @@
* {
-GtkDialog-action-area-border: 6;
}
button {
min-height: 24px;
min-width: 24px;
}
entry {
min-height: 24px;
-GtkDialog-action-area-border: 12;
}
switch {
margin-right: 2px;
margin-right: 2px;
}
spinbutton entry {
min-height: 16px;
}

View File

@@ -1,5 +1,5 @@
#!/bin/bash
VER="2.1.2_Beta"
VER="2.2.0_Beta"
B_PATH="dist/DemonEditor"
DEB_PATH="$B_PATH/usr/share/demoneditor"

View File

@@ -1,5 +1,5 @@
Package: demon-editor
Version: 2.1.2-Beta
Version: 2.2.0-Beta
Section: utils
Priority: optional
Architecture: all

View File

@@ -5,6 +5,8 @@ Comment=Channel and satellite list editor for Enigma2
Comment[ru]=Редактор списка каналов и спутников для Enigma2
Comment[be]=Рэдактар спіса каналаў і спадарожнікаў для Enigma2
Comment[de]=Programm- und Satellitenlisten-Editor für Enigma2
Comment[tr]=Enigma2 için kanal ve uydu listesi editörü
Comment[es]=Editor de listas de canales y satélites para Enigma2
Icon=demon-editor
Exec=/usr/bin/demon-editor
Terminal=false

View File

@@ -69,7 +69,7 @@ app = BUNDLE(coll,
'CFBundleGetInfoString': "Enigma2 channel and satellite editor",
'LSApplicationCategoryType': 'public.app-category.utilities',
'LSMinimumSystemVersion': '10.13',
'CFBundleShortVersionString': f"2.1.2.{BUILD_DATE} Beta",
'CFBundleShortVersionString': f"2.2.0.{BUILD_DATE} Beta",
'NSHumanReadableCopyright': u"Copyright © 2022, Dmitriy Yefremov",
'NSRequiresAquaSystemAppearance': 'false'
})

View File

@@ -1333,3 +1333,18 @@ msgstr "Зменена месцаванне элементаў!"
msgid "Restart the program to apply all changes."
msgstr "Перазапусціце праграму, каб ужыць усе змены."
msgid "New folder"
msgstr "Стварыць тэчку"
msgid "Rename"
msgstr "Змяніць назву"
msgid "Bookmarks"
msgstr "Закладкі"
msgid "Add bookmark"
msgstr "Дадаць закладку"
msgid "All bouquets"
msgstr "Усе букеты"

View File

@@ -1347,3 +1347,19 @@ msgstr "Das Layout der Elemente wurde geändert!"
msgid "Restart the program to apply all changes."
msgstr "Starte das Programm neu, um alle Änderungen zu übernehmen."
msgid "New folder"
msgstr "Ordner erstellen"
msgid "Rename"
msgstr "Umbenennen"
msgid "Bookmarks"
msgstr "Lesezeichen"
msgid "Add bookmark"
msgstr "Lesezeichen hinzufügen"
msgid "All bouquets"
msgstr "Alle Bouquets"

View File

@@ -1,23 +1,25 @@
# Copyright (C) 2018-2020 Frank Neirynck
# Copyright (C) 2018-2022 Víctor Pont
# This file is distributed under the MIT license.
#
# Frank Neirynck <frank@insink.be>, 2018-2020.
#
# Víctor Pont <victor@pont.cat>, 2021-2022.
msgid ""
msgstr ""
"Last-Translator: Frank Neirynck\n"
"Project-Id-Version: \n"
"POT-Creation-Date: \n"
"PO-Revision-Date: \n"
"Last-Translator: Víctor Pont\n"
"Language-Team: \n"
"Language: es\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Project-Id-Version: \n"
"POT-Creation-Date: \n"
"PO-Revision-Date: \n"
"Language-Team: \n"
"X-Generator: Poedit 2.2.1\n"
"X-Generator: Poedit 3.0.1\n"
msgid "translator-credits"
msgstr "Frank Neirynck <frank@insink.be>"
msgstr ""
"Víctor Pont <victor@pont.cat>\n"
"Frank Neirynck <frank@insink.be>"
# Main
msgid "Service"
@@ -33,19 +35,19 @@ msgid "Picon"
msgstr "Picon"
msgid "Freq"
msgstr "Frec."
msgstr "Frec"
msgid "Rate"
msgstr "Ratio"
msgid "Pol"
msgstr "Pol."
msgstr "Pol"
msgid "System"
msgstr "Sistema"
msgid "Pos"
msgstr "Pos."
msgstr "Pos"
msgid "Num"
msgstr "Núm"
@@ -56,6 +58,9 @@ msgstr "IP actual:"
msgid "Assign"
msgstr "Asignar"
msgid "Assign file"
msgstr "Asignar fichero"
msgid "Bouquet details"
msgstr "Detalles bouquet"
@@ -110,6 +115,9 @@ msgstr "Establecer nombre predeterminado"
msgid "Insert marker"
msgstr "Insertar marcador"
msgid "Insert space"
msgstr "Insertar espacio"
msgid "Locate in services"
msgstr "Buscar en servicios"
@@ -162,10 +170,10 @@ msgid "Satellites downloader"
msgstr "Descarga de satélites"
msgid "Remove"
msgstr "Quitar"
msgstr "Borrar"
msgid "Remove all unavailable"
msgstr "Quitar todo lo indisponible"
msgstr "Borrar lo que no esté disponible"
msgid "Satellites editor"
msgstr "Editor de satélites"
@@ -225,7 +233,7 @@ msgid "Receiver IP:"
msgstr "IP del receptor:"
msgid "Remove unused bouquets"
msgstr "Quitar bouquets sin usar"
msgstr "Borrar bouquets sin usar"
msgid "Reset profile"
msgstr "Restablecer perfil"
@@ -233,6 +241,9 @@ msgstr "Restablecer perfil"
msgid "Satellites"
msgstr "Satélites"
msgid "Transponders"
msgstr "Transponders"
msgid "Satellites.xml file:"
msgstr "Fichero satellites.xml:"
@@ -327,7 +338,7 @@ msgid "Path to Enigma2 picons:"
msgstr "Ruta a picons de Enigma2:"
msgid "Specify the correct position value for the provider!"
msgstr "¡Especifique el valor correcto de la posición del proveedor!"
msgstr "¡Especifica el valor correcto de la posición del proveedor!"
msgid "Converter between name formats"
msgstr "Conversor entre formatos de nombre"
@@ -341,10 +352,9 @@ msgstr "Cargar proveedores de satélite."
msgid ""
"To automatically set the identifiers for picons,\n"
"first load the required services list into the main application window."
msgstr ""
"Para configurar automáticamente los identificadores para picons,\n"
"cargue primero la lista de servicios requeridos en la ventana principal."
"carga primero la lista de servicios requeridos en la ventana principal."
# Satellites editor
msgid "Satellites edit tool"
@@ -376,7 +386,7 @@ msgid "Satellites update"
msgstr "Actualizar satélites"
msgid "Remove selection"
msgstr "Quitar selección"
msgstr "Borrar selección"
# Service details dialog
msgid "Service data:"
@@ -396,7 +406,7 @@ msgid ""
"Continue?"
msgstr ""
"Los cambios se aplicarán a todos los servicios de este transpondedor!\n"
"¿Continuar?"
"¿Quieres continuar?"
msgid "Reference"
msgstr "Referencia"
@@ -408,7 +418,7 @@ msgid "Flags:"
msgstr "Flags:"
msgid "Delays (ms):"
msgstr "Retraso (ms)"
msgstr "Retraso (ms):"
msgid "Bitstream"
msgstr "Secuencia de bits"
@@ -493,7 +503,7 @@ msgid "Error. No bouquet is selected!"
msgstr "Error. ¡Ningún bouquet seleccionado!"
msgid "This item is not allowed to be removed!"
msgstr "¡Este elemento no puede ser quitado!"
msgstr "¡Este elemento no puede ser borrado!"
msgid "This item is not allowed to edit!"
msgstr "¡Este elemento no puede ser editado!"
@@ -502,7 +512,7 @@ msgid "Not allowed in this context!"
msgstr "¡No permitido en este contexto!"
msgid "Please, download files from receiver or setup your path for read data!"
msgstr "Por favor, descargue ficheros desde el receptor o configure la ruta para leer los datos!"
msgstr "Por favor, descarga ficheros desde el receptor o configura la ruta para leer los datos!"
msgid "Reading data error!"
msgstr "¡Error de lectura de datos!"
@@ -511,17 +521,20 @@ msgid "No m3u file is selected!"
msgstr "¡No se ha seleccionado ningún fichero m3u!"
msgid "Not implemented yet!"
msgstr "¡No implementado!"
msgstr "¡No implementado todavía!"
msgid "The text of marker is empty, please try again!"
msgstr "¡El texto del marcador está vacío, inténtalo de nuevo!"
msgid "Please, select only one item!"
msgstr "¡Por favor, seleccione un único elemento!"
msgstr "¡Por favor, selecciona un único elemento!"
msgid "No png file is selected!"
msgstr "¡No se ha seleccionado ningún fichero png!"
msgid "No profile selected!"
msgstr "¡No se ha seleccionado ningún perfil!"
msgid "No reference is present!"
msgstr "¡Ninguna referencia presente!"
@@ -535,7 +548,7 @@ msgid "Done!"
msgstr "¡Hecho!"
msgid "Please, wait..."
msgstr "Por favor, espere..."
msgstr "Por favor, espera..."
msgid "Resizing..."
msgstr "Redimensionando..."
@@ -550,29 +563,29 @@ msgid "Please, select only one satellite!"
msgstr "¡Seleccione sólo un Satélite!"
msgid "Please check your parameters and try again."
msgstr "¡Por favor revise sus parámetros y vuelva a intentarlo!"
msgstr "Por favor, revisa tus parámetros y vuelve a intentarlo."
msgid "No satellites.xml file is selected!"
msgstr "¡Ningún satellites.xml seleccionado!"
msgid "Error. Verify the data!"
msgstr "Error. ¡Revise los datos!"
msgstr "Error. ¡Revisa los datos!"
msgid "Operation not allowed in this context!"
msgstr "¡Operación no permitida en este contexto!"
msgid "No VLC is found. Check that it is installed!"
msgstr "VLC no encontrado. ¡Compruebe que está instalado!"
msgstr "VLC no encontrado. ¡Comprueba que está instalado!"
# Search unavailable streams dialog
msgid "Please wait, streams testing in progress..."
msgstr "Por favor espere, hay una prueba de stream en progreso..."
msgstr "Por favor espera, hay una prueba de stream en progreso..."
msgid "Found"
msgstr "Encontrado"
msgstr "Encontrado(s)"
msgid "unavailable streams."
msgstr "Streams no presentes."
msgstr "streams no presentes."
msgid "No changes required!"
msgstr "¡No se requieren cambios!"
@@ -657,10 +670,10 @@ msgid "No bouquet file is selected!"
msgstr "¡No se ha seleccionado nigún fichero de bouquet!"
msgid "Remove all unused"
msgstr "Quitar todos sin usar"
msgstr "Borrar los que están sin usar"
msgid "Test"
msgstr "Prueba"
msgstr "Probar"
msgid "Test connection"
msgstr "Probar conexión"
@@ -684,7 +697,7 @@ msgid "Enable HTTP API"
msgstr "Habilitar API HTTP"
msgid "Switch(zap) the channel(Ctrl + Z)"
msgstr "Poner el canal (Ctrl + Z)"
msgstr "Poner (zapear) el canal (Ctrl + Z)"
msgid "Switch the channel and watch in the program(Ctrl + W)"
msgstr "Poner el canal y ver en el programa (Ctrl + W)"
@@ -708,7 +721,7 @@ msgid "Service names source:"
msgstr "Origen nombres de servicio:"
msgid "Main service list"
msgstr "Lista principal de servicios:"
msgstr "Lista principal de servicios"
msgid "XML file"
msgstr "Fichero XML"
@@ -770,8 +783,12 @@ msgstr "Terminar reproducción"
msgid "Import YouTube playlist"
msgstr "Importar lista de reproducción de YouTube"
msgid "Found a link to the YouTube resource!\nTry to get a direct link to the video?"
msgstr "¡Se ha encontrado enlace al recurso de YouTube!\n¿Intentar obtener un enlace directo al vídeo?"
msgid ""
"Found a link to the YouTube resource!\n"
"Try to get a direct link to the video?"
msgstr ""
"¡Se ha encontrado un enlace al recurso de YouTube!\n"
"¿Quieres intentar obtener un enlace directo al vídeo?"
msgid "Playlist import"
msgstr "Importar lista de reproducción"
@@ -786,7 +803,7 @@ msgid "Apply profile settings"
msgstr "Aplicar ajustes de perfil"
msgid "Settings type:"
msgstr "Tipo de ajustes:"
msgstr "Tipo:"
msgid "Set default"
msgstr "Por defecto"
@@ -813,7 +830,7 @@ msgid "Drag or paste the link here"
msgstr "Soltar o pegar en enlace aquí"
msgid "Remove added links in the playlist"
msgstr "Quitar los enlaces añadidos en la lista de reproducción"
msgstr "Borrar los enlaces añadidos a la lista de reproducción"
msgid "A bouquet with that name exists!"
msgstr "¡Ya existe un bouquet con ese nombre!"
@@ -867,7 +884,7 @@ msgid "IPTV tools"
msgstr "Intrumentos IPTV"
msgid "Make profile folder as default for the additional data"
msgstr "Usar por defecto el directorio de perfil para los datos adionales"
msgstr "Usar por defecto el directorio de perfil para los datos adicionales"
msgid "Default data path:"
msgstr "Ruta estándar de datos:"
@@ -923,8 +940,8 @@ msgstr "Modo de reproducción de los streams:"
msgid "Built-in player"
msgstr "Reproductor interno"
msgid "VLC media player"
msgstr "Reproductor VLC"
msgid "In a separate window"
msgstr "En una ventana aparte"
msgid "Only get m3u file"
msgstr "Sólo bajar fichero m3u"
@@ -933,7 +950,7 @@ msgid "Save and restart the program to apply the settings."
msgstr "Guardar y reiniciar el programa para aplicar la configuración."
msgid "Some images may have problems displaying the favorites list!"
msgstr "Algunas imágenes pueden tener problemas al mostrar la lista de favoritos!"
msgstr "¡Algunas imágenes pueden tener problemas al mostrar la lista de favoritos!"
msgid "Operates in standby mode or current active transponder!"
msgstr "¡Funciona en modo de espera o transpondedor activo actual!"
@@ -984,7 +1001,7 @@ msgid "Download from the receiver"
msgstr "Descargar desde el receptor"
msgid "Remove all picons from the receiver"
msgstr "Eliminar todos los picons del receptor"
msgstr "Borrar todos los picons del receptor"
msgid "Service reference"
msgstr "Referencia del servicio"
@@ -1010,14 +1027,24 @@ msgstr "¡EXPERIMENTAL!"
msgid "Sorting data..."
msgstr "Ordenando datos..."
msgid "There are unsaved changes.\n\n\t Save them now?"
msgstr "Hay cambios sin guardar.\n\n\t ¿Desea guardarlos ahora?"
msgid ""
"There are unsaved changes.\n"
"\n"
"\t Save them now?"
msgstr ""
"Hay cambios sin guardar.\n"
"\n"
"\t ¿Desea guardarlos ahora?"
msgid "Are you sure you want to change the order\n\t of services in this bouquet?"
msgstr "¿Está seguro de querer cambiar el orden\n\t de servicios en este bouquet?"
msgid ""
"Are you sure you want to change the order\n"
"\t of services in this bouquet?"
msgstr ""
"¿Está seguro de querer cambiar el orden\n"
"\t de servicios en este bouquet?"
msgid "Remove from the receiver"
msgstr "Eliminar del receptor"
msgstr "Borrar del receptor"
msgid "Screenshot"
msgstr "Captura de pantalla"
@@ -1025,3 +1052,333 @@ msgstr "Captura de pantalla"
msgid "Video"
msgstr "Vídео"
msgid "The Neutrino has only experimental support. Not all features are supported!"
msgstr "El Neutrino sólo dispone de soporte experimental. ¡No todas las funciones estarán disponibles!"
msgid "Enable experimental features"
msgstr "Activar las funciones experimentales"
msgid "Can't Playback!"
msgstr "¡No se puede reproducir!"
msgid "Enable Dark Mode"
msgstr "Habilitar modo oscuro"
msgid "Extract..."
msgstr "Extraer..."
msgid "Unsupported format!"
msgstr "¡Formato no soportado!"
msgid "Combine with the current data?"
msgstr "¿Combinar con los datos actuales?"
msgid "Importing data done!"
msgstr "¡Importación finalizada!"
msgid "Current service"
msgstr "Servicio actual"
msgid "Open folder"
msgstr "Abrir carpeta"
msgid "Open archive"
msgstr "Abrir fichero"
msgid "Import from Web"
msgstr "Importar de Internet"
msgid "Control"
msgstr "Control"
msgid "Timers"
msgstr "Temporizadores"
msgid "Timer"
msgstr "Temporizador"
msgid "Add timer"
msgstr "Añadir temporizador"
msgid "Hr."
msgstr "Hora"
msgid "Min."
msgstr "Min."
msgid "Power"
msgstr "Power"
msgid "Standby"
msgstr "Standby"
msgid "Wake Up"
msgstr "Despertar"
msgid "Reboot"
msgstr "Reiniciar"
msgid "Restart GUI"
msgstr "Reiniciar GUI"
msgid "Shutdown"
msgstr "Apagar"
msgid "Shut down"
msgstr "Apagar"
msgid "Do Nothing"
msgstr "No hacer nada"
msgid "Auto"
msgstr "Auto"
msgid "Grab screenshot"
msgstr "Captura de pantalla"
msgid "Enabled:"
msgstr "Activo:"
msgid "Name:"
msgstr "Nombre:"
msgid "Description:"
msgstr "Descripción:"
msgid "Service:"
msgstr "Servicio:"
msgid "Service reference:"
msgstr "Referencia del servicio:"
msgid "Event ID:"
msgstr "ID evento:"
msgid "Begins:"
msgstr "Inicio:"
msgid "Ends:"
msgstr "Final:"
msgid "Repeated:"
msgstr "Repetir:"
msgid "Action:"
msgstr "Acción:"
msgid "After event:"
msgstr "Después del evento:"
msgid "Location:"
msgstr "Ubicación:"
msgid "Mo"
msgstr "Lu"
msgid "Tu"
msgstr "Ma"
msgid "We"
msgstr "Mi"
msgid "Th"
msgstr "Ju"
msgid "Fr"
msgstr "Vi"
msgid "Sa"
msgstr "Sa"
msgid "Su"
msgstr "Do"
msgid "Set"
msgstr "Poner"
msgid "Services update"
msgstr "Actualización servicios"
msgid "Create folder"
msgstr "Crear carpeta"
msgid "FTP client"
msgstr "Cliente FTP"
msgid "The file size is too large!"
msgstr "¡El fichero es demasiado grande!"
msgid "Connect"
msgstr "Conectar"
msgid "Disconnect"
msgstr "Desconectar"
msgid "Size"
msgstr "Tamaño"
msgid "Date"
msgstr "Fecha"
msgid "Attr."
msgstr "Atr."
msgid "Toggle display position"
msgstr "Cambiar la posición de la pantalla"
msgid "Alternatives"
msgstr "Alternativas"
msgid "Add alternatives"
msgstr "Añadir alternativas"
msgid "DreamOS only!"
msgstr "¡Sólo DreamOS!"
msgid "A similar service is already in this list!"
msgstr "¡Un servicio similar ya está en esta lista!"
msgid ""
"Play mode has been changed!\n"
"Restart the program to apply the settings."
msgstr ""
"¡El modo de reproducción se ha cambiado!\n"
"Reinicia el programa para aplicar los cambios."
msgid "Set values for TID, NID and Namespace for correct naming of the picons!"
msgstr "¡Poner los valores de TID, NID y Namespace para una correcta nomenclatura de los picons!"
msgid "Streams detected:"
msgstr "Streams encontrados:"
msgid "Download picons"
msgstr "Descargar picons"
msgid "Errors:"
msgstr "Errores:"
msgid "Use to play streams:"
msgstr "Reproductor de streams:"
msgid "Font in the lists:"
msgstr "Fuente en las listas:"
msgid "Picons size in the lists:"
msgstr "Tamaño de los picons en las listas:"
msgid "Logo size in tooltips:"
msgstr "Tamaño de los logos en las listas:"
msgid "Save as"
msgstr "Guardar como"
msgid "Mark duplicates"
msgstr "Marcar duplicados"
msgid "Load only for selected bouquet"
msgstr "Cargar sólo para el bouquet seleccionado"
msgid "The task is canceled!"
msgstr "¡Se ha cancelado la tarea!"
msgid "Data loading in progress!"
msgstr "¡Carga de datos en progreso!"
msgid "Recordings"
msgstr "Grabaciones"
msgid "Help"
msgstr "Ayuda"
msgid "HTTP API is not activated. Check your settings!"
msgstr "HTTP API no activa. ¡Comprueba la configuración!"
msgid "Add picons"
msgstr "Añadir picons"
msgid "Logs"
msgstr "Logs"
msgid "Title"
msgstr "Título"
msgid "Time"
msgstr "Hora"
msgid "Length"
msgstr "Duración"
msgid "Additional source"
msgstr "Fuente adicional"
msgid "Automatically set the name selected in the favorites list."
msgstr "Poner automáticamente el nombre en la lista de favoritos seleccionada."
msgid "Playback"
msgstr "Reproducción"
msgid "Audio"
msgstr "Audio"
msgid "Audio Track"
msgstr "Pista de audio"
msgid "Subtitle"
msgstr "Subtítulos"
msgid "Subtitle Track"
msgstr "Pista de subtítulos"
msgid "Aspect ratio"
msgstr "Relación de aspecto"
msgid "This may change the settings of other profiles!"
msgstr "¡Esto podría cambiar la configuración de otros perfiles!"
msgid "Drag the services to the desired picon or picon to the list of selected services."
msgstr "Arrastrar los servicios al picon deseado o al picon de la lista de los servicios seleccionados."
msgid "Sets the profile folder as default to store picons, backups, etc."
msgstr "Pone la carpeta del perfil como por defecto para almacenar picons, backups, etc."
msgid "New sub-bouquet"
msgstr "Nuevo sub-bouquet"
msgid "Mark not presented in Bouquets"
msgstr "Marcar los no presentes en los bouquets"
msgid "Not in Bouquets"
msgstr "No en los bouquets"
msgid "Do not show services present in Bouquets."
msgstr "No mostrar los servicios presentes en los bouquets."
msgid "IPTV services only"
msgstr "Sólo servicios IPTV"
msgid "Display picons"
msgstr "Mostrar picons"
msgid "Alternate layout"
msgstr "Diseño alternativo"
msgid "Layout of elements has been changed!"
msgstr "¡El diseño de los elementos ha cambiado!"
msgid "Restart the program to apply all changes."
msgstr "Reinicia el programa para aplicar los cambios."
msgid "New folder"
msgstr "Nueva carpeta"
msgid "Rename"
msgstr "Renombrar"
msgid "Bookmarks"
msgstr "Marcadores"
msgid "Add bookmark"
msgstr "Añadir marcador"
msgid "All bouquets"
msgstr "Todos los bouquets"

View File

@@ -1,4 +1,4 @@
# Copyright (C) 2018-2021 Dmitriy Yefremov
# Copyright (C) 2018-2022 Dmitriy Yefremov
# This file is distributed under the MIT license.
#
#
@@ -1330,3 +1330,18 @@ msgstr "Изменено расположение элементов!"
msgid "Restart the program to apply all changes."
msgstr "Перезапустите программу, чтобы применить все изменения."
msgid "New folder"
msgstr "Создать папку"
msgid "Rename"
msgstr "Переименовать"
msgid "Bookmarks"
msgstr "Закладки"
msgid "Add bookmark"
msgstr "Добавить в закладки"
msgid "All bouquets"
msgstr "Все букеты"

View File

@@ -3,7 +3,7 @@ msgstr ""
"Project-Id-Version: DemonEditor\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2020-04-16 15:59+0300\n"
"PO-Revision-Date: 2021-11-11 23:49+0300\n"
"PO-Revision-Date: 2022-02-21 22:06+0300\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
@@ -53,6 +53,9 @@ msgstr "Geçerli IP:"
msgid "Assign"
msgstr "Ata"
msgid "Assign file"
msgstr "Dosya ata"
msgid "Bouquet details"
msgstr "Buket detayları"
@@ -233,6 +236,9 @@ msgstr "Profili sıfırla"
msgid "Satellites"
msgstr "Uydular"
msgid "Transponders"
msgstr "Transponders"
msgid "Satellites.xml file:"
msgstr "Satellites.xml dosyası:"
@@ -1329,3 +1335,30 @@ msgstr "Hizmetleri istediğiniz simgeye veya simgeyi seçili hizmetler listesine
msgid "Sets the profile folder as default to store picons, backups, etc."
msgstr "Picon'ları, yedekleri vb. depolamak için profil klasörünü varsayılan olarak ayarlar."
msgid "New sub-bouquet"
msgstr "Yeni alt buket"
msgid "Mark not presented in Bouquets"
msgstr "Buketlerde olmayanları işaretleyin"
msgid "Not in Bouquets"
msgstr "Buketlerde Değil"
msgid "Do not show services present in Bouquets."
msgstr "Buketlerde bulunan hizmetleri göstermeyin."
msgid "IPTV services only"
msgstr "Yalnızca IPTV hizmetleri"
msgid "Display picons"
msgstr "Display piconlar"
msgid "Alternate layout"
msgstr "Alternatif düzen"
msgid "Layout of elements has been changed!"
msgstr "Öğelerin düzeni değiştirildi!"
msgid "Restart the program to apply all changes."
msgstr "Tüm değişiklikleri uygulamak için programı yeniden başlatın."