aboutsummaryrefslogtreecommitdiff
path: root/src/CurlyTx.py
blob: 3e74c5e54e9fbfdcb1cd3d97e4803b1e36f7a482 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
# -*- coding: utf-8 -*-
# CurlyTx main window
# Copyright (C) 2011 Christian Weiske <cweiske@cweiske.de>
# License: GPLv3 or later

from . import _

from Screens.Screen import Screen
from Screens.HelpMenu import HelpableScreen
from Components.Label import Label
from Components.ScrollLabel import ScrollLabel
from Components.ActionMap import ActionMap
from Components.Sources.StaticText import StaticText
from Tools import Directories
from twisted.web import client
from twisted.web.client import _makeGetterFactory, HTTPClientFactory
from enigma import gFont

from . import config
from config import createPage, loadDefaultPageOptions, feedPagesToConfig, feedSettingsToConfig, savePageConfig
from Components.config import config

import os

class CurlyTx(Screen, HelpableScreen):
    skin = """
        <screen name="CurlyTx" position="center,center" size="560,430" title="CurlyTx" >
	  <ePixmap position="0,0" size="140,40" pixmap="skin_default/buttons/red.png" transparent="1" alphatest="on" />
	  <ePixmap position="140,0" size="140,40" pixmap="skin_default/buttons/green.png" transparent="1" alphatest="on" />
	  <ePixmap position="280,0" size="140,40" pixmap="skin_default/buttons/yellow.png" transparent="1" alphatest="on" />
	  <ePixmap position="420,0" size="140,40" pixmap="skin_default/buttons/blue.png" transparent="1" alphatest="on" />
	  <widget source="key_red" render="Label" position="0,0" zPosition="1" size="140,40" valign="center" halign="center" font="Regular;21" transparent="1" foregroundColor="white" />
	  <widget source="key_green" render="Label" position="140,0" zPosition="1" size="140,40" valign="center" halign="center" font="Regular;21" transparent="1" foregroundColor="white" />
	  <widget source="key_yellow" render="Label" position="280,0" zPosition="1" size="140,40" valign="center" halign="center" font="Regular;21" transparent="1" foregroundColor="white" />
	  <widget source="key_blue" render="Label" position="420,0" zPosition="1" size="140,40" valign="center" halign="center" font="Regular;21" transparent="1" foregroundColor="white" />
	  <widget name="text" position="5,45" size="550,380" font="Console;20" />
        </screen>"""

    currentUrl = None
    currentPage = None
    currentFontSize = 20
    httpGetterFactory = None
    showingHeaders = False
    staticPageFeedFile = 'curlytx-pagefeed.xml'

    def __init__(self, session, args = None):
        #self.skin = CurlyTx.skin
        Screen.__init__(self, session)
        HelpableScreen.__init__(self)
        #self.skinName = [ "CurlyTx", "Setup" ]

        self["text"] = ScrollLabel("foo")

        self["key_red"]    = StaticText(_("Settings"))
        self["key_green"]  = StaticText(_("Reload"))
        self["key_yellow"] = StaticText(_("Prev"))
        self["key_blue"]   = StaticText(_("Next"))


        self["actions"] = ActionMap(
            ["WizardActions", "ColorActions", "InputActions", "InfobarEPGActions"], {
                "ok":   self.close,
                "back": self.close,
                "up":   self.pageUp,
                "down": self.pageDown,

                "red":    self.showSettings,
                "green":  self.reload,
                "yellow": self.prevPage,
                "blue":   self.nextPage,

                "showEventInfo": self.showHeader
            }, -1)

        self.loadHelp()
        self.loadButtons()
        self.onFirstExecBegin.append(self.afterLayout)

    def afterLayout(self):
        self.setTextFont
        self.loadStaticConfig()
        self.loadUrl(config.plugins.CurlyTx.defaultPage.value)

    def loadHelp(self):
        self.helpList.append((
                self["actions"], "WizardActions",
                [("up", _("Scroll page contents up"))]))
        self.helpList.append((
                self["actions"], "WizardActions",
                [("down", _("Scroll page contents down"))]))
        self.helpList.append((
                self["actions"], "InfobarEPGActions",
                [("showEventInfo", _("Show HTTP response headers"))]))
        self.helpList.append((
                self["actions"], "ColorActions",
                [("red", _("Show program settings"))]))
        self.helpList.append((
                self["actions"], "ColorActions",
                [("green", _("Reload current page URL"))]))
        self.helpList.append((
                self["actions"], "ColorActions",
                [("yellow", _("Switch to next configured page URL"))]))
        self.helpList.append((
                self["actions"], "ColorActions",
                [("blue", _("Switch to previous configured page URL"))]))
        self.helpList.append((
                self["actions"], "WizardActions",
                [("ok", _("Close window"))]))
        self.helpList.append((
                self["actions"], "WizardActions",
                [("back", _("Close window"))]))

    def loadButtons(self):
        pageCount = len(config.plugins.CurlyTx.pages)
        if pageCount == 0:
            self["key_green"].setText("")
            self["key_yellow"].setText("")
            self["key_blue"].setText("")
        elif pageCount == 1:
            self["key_green"].setText(_("Reload"))
            self["key_yellow"].setText("")
            self["key_blue"].setText("")
        else:
            self["key_green"].setText(_("Reload"))
            self["key_yellow"].setText(_("Prev"))
            self["key_blue"].setText(_("Next"))

        if config.plugins.CurlyTx.enableSettings.getValue():
            self["key_red"].setText(_("Settings"))
        else:
            self["key_red"].setText("")

    def pageUp(self):
        self["text"].pageUp()

    def pageDown(self):
        self["text"].pageDown()

    def prevPage(self):
        if self.currentPage == None:
            return

        pageId = self.currentPage - 1
        if pageId < 0:
            pageId = len(config.plugins.CurlyTx.pages) - 1
        self.loadUrl(pageId)

    def nextPage(self):
        if self.currentPage == None:
            return

        pageId = self.currentPage + 1
        if pageId > len(config.plugins.CurlyTx.pages) - 1:
            pageId = 0
        self.loadUrl(pageId)

    def reload(self):
        if self.currentPage == None:
            return

        self.loadUrl(self.currentPage)

    def loadUrl(self, pageId):
        self.httpGetterFactory = None
        if pageId == None:
            self.loadNoPage()
            return

        cfg = config.plugins.CurlyTx
        pageCount = len(cfg.pages)
        pageId = int(pageId)
        if pageId > (pageCount - 1):
            if len(cfg.pages) == 0:
                self.loadNoPage()
            else:
                self["text"].setText(_("Invalid page") + " " + pageId);
            return

        url   = cfg.pages[pageId].uri.value
        title = cfg.pages[pageId].title.value

        if pageCount > 1:
            title = "{0} [{1}/{2}]".format(title, pageId + 1, pageCount)

        self.currentPage = pageId
        self.currentUrl = url
        self.currentFontSize = cfg.pages[pageId].fontSize.value

        self.setTitle(title)
        self.setTextFont()
        self["text"].setText(_("Loading ...") + "\n" + url);

        if (url.startswith('file://')):
            self.loadLocalFile(url)
        else:
            self.getPageWebClient(url).addCallback(self.urlLoaded).addErrback(self.urlFailed, url)

    def loadLocalFile(self, url):
        file = url[7:]
        if not os.path.exists(file):
            self.showFail('File does not exist', file)
            return

        with open(file, 'r') as f:
            self.urlLoaded(f.read())

    def setTextFont(self):
        if self["text"].long_text is not None:
            self["text"].long_text.setFont(gFont("Console", self.currentFontSize))

    def urlLoaded(self, html):
        self["text"].setText(html)

    def urlFailed(self, error, url):
        self.showFail(error.getErrorMessage(), url)

    def showFail(self, message, url):
        self["text"].setText(
            _("Error fetching URL:") + "\n " + message
            + "\n\nURL: " + url
            )

    def loadNoPage(self):
        self["text"].setText(_("Go and add a page in the settings"));

    def showHeader(self):
        if not self.httpGetterFactory:
            return

        if self.showingHeaders:
            self["text"].setText(self.pageContent)
            self.pageContent    = None
            self.showingHeaders = False
        elif self.httpGetterFactory.response_headers:
            headers = _("HTTP response headers for") + "\n" + self.currentUrl + "\n\n"
            for (k, v) in self.httpGetterFactory.response_headers.items():
                headers += k + ": " + ("\n" + k + ": ").join(v) + "\n"
            self.pageContent = self["text"].getText()
            self["text"].setText(headers)
            self.showingHeaders = True

    def showSettings(self):
        if not config.plugins.CurlyTx.enableSettings.getValue():
            return

        from CurlyTxSettings import CurlyTxSettings
        self.session.openWithCallback(self.onSettingsChanged, CurlyTxSettings)

    def onSettingsChanged(self):
        self.loadButtons()
        if len(config.plugins.CurlyTx.pages) == 0:
            self.currentPage = None
            self.loadUrl(self.currentPage)
        elif self.currentPage == None:
            self.currentPage = 0
            self.loadUrl(self.currentPage)


    def getPageWebClient(self, url, contextFactory=None, *args, **kwargs):
        """
        Download a web page as a string.

        COPY OF twisted.web.client.getPage to store the factory

        Download a page. Return a deferred, which will callback with a
        page (as a string) or errback with a description of the error.

        See L{HTTPClientFactory} to see what extra arguments can be passed.
        """
        self.httpGetterFactory = _makeGetterFactory(
            url,
            HTTPClientFactory,
            contextFactory=contextFactory,
            *args, **kwargs)
        return self.httpGetterFactory.deferred

    def loadStaticConfig(self):
        """
        Always try to load the static config file from
        /etc/enigma2/curlytx-pagefeed.xml
        """
        staticFeedPath = Directories.resolveFilename(Directories.SCOPE_CONFIG, self.staticPageFeedFile)
        if not os.path.exists(staticFeedPath):
            return

        from AtomFeed import AtomFeed
        AtomFeed(
            'file://' + staticFeedPath,
            self.saveStaticConfig, self.loadStaticConfigFail
            )

    def loadStaticConfigFail(self, errorMessage):
        """ Loading the page url feed failed somehow """
        from Screens.MessageBox import MessageBox
        self.session.open(
            MessageBox,
            _("Error loading page feed:") + "\n\n" + str(errorMessage),
            type = MessageBox.TYPE_ERROR
            )

    def saveStaticConfig(self, pages, settings):
        feedPagesToConfig(pages)
        feedSettingsToConfig(settings)
        savePageConfig()
        self.loadButtons()