From db051fcf34338e31748b1a197878bb6bd33d84a0 Mon Sep 17 00:00:00 2001 From: Jan Kowalleck Date: Thu, 10 Oct 2019 21:14:16 +0200 Subject: [PATCH 01/39] 2.0-dev cleanup --- .editorconfig | 34 - .gitignore | 56 -- .gitmodules | 6 - README.md | 9 + config.defaults.ini | 34 - configs/demo.ini | 15 - configs/kadsen.ini | 9 - configs/nsfw.ini | 15 - configs/pr0.ini | 9 - configs/sfw.ini | 14 - crawler/__init__.py | 442 ------------- crawler/fourchan.py | 59 -- crawler/giphy.py | 52 -- crawler/instagram.py | 65 -- crawler/ninegag.py | 78 --- crawler/pr0gramm.py | 105 --- crawler/reddit.py | 55 -- crawler/soupio.py | 63 -- nichtparasoup.py | 402 ------------ requirements.txt | 9 - templates.py | 374 ----------- templates_raw/README.md | 26 - templates_raw/_bundler | 1 - templates_raw/_foreign/normalize.css | 1 - templates_raw/build.sh | 34 - templates_raw/root/.editorconfig | 36 -- templates_raw/root/README.md | 26 - templates_raw/root/build.sh | 10 - templates_raw/root/css/feel.css | 11 - templates_raw/root/css/look.css | 200 ------ templates_raw/root/css/look_bossMode.css | 7 - templates_raw/root/css/look_buttons.css | 43 -- templates_raw/root/css/look_hotkeys.css | 60 -- templates_raw/root/css/look_stateSwitch.css | 24 - templates_raw/root/css/normalize.css | 1 - templates_raw/root/css/sourceIcons.css | 43 -- templates_raw/root/flush | 1 - templates_raw/root/get | 7 - templates_raw/root/js/feel.js | 485 -------------- templates_raw/root/js/feel_hotkeys.js | 122 ---- templates_raw/root/js/helper.js | 71 --- templates_raw/root/js/look_maxSizer.js | 116 ---- templates_raw/root/js/look_stateSwitch.js | 25 - templates_raw/root/js/snowstorm.js | 668 -------------------- templates_raw/root/reset | 1 - templates_raw/root/root.html | 163 ----- templates_raw/root/test/get_local | 7 - templates_raw/root/test/get_remote | 7 - templates_raw/root/test/testimg.png | Bin 1761 -> 0 bytes tests/configs/README.md | 10 - tests/configs/factors.ini | 11 - tests/configs/fourchan.ini | 10 - tests/configs/giphy.ini | 9 - tests/configs/instagram.ini | 10 - tests/configs/ninegag.ini | 9 - tests/configs/pr0gramm.ini | 9 - tests/configs/reddit.ini | 9 - tests/configs/soup.ini | 9 - tests/logs/.gitignore | 3 - 59 files changed, 9 insertions(+), 4181 deletions(-) delete mode 100644 .editorconfig delete mode 100644 .gitignore delete mode 100644 .gitmodules delete mode 100644 config.defaults.ini delete mode 100644 configs/demo.ini delete mode 100644 configs/kadsen.ini delete mode 100644 configs/nsfw.ini delete mode 100644 configs/pr0.ini delete mode 100644 configs/sfw.ini delete mode 100644 crawler/__init__.py delete mode 100644 crawler/fourchan.py delete mode 100644 crawler/giphy.py delete mode 100644 crawler/instagram.py delete mode 100644 crawler/ninegag.py delete mode 100644 crawler/pr0gramm.py delete mode 100644 crawler/reddit.py delete mode 100644 crawler/soupio.py delete mode 100755 nichtparasoup.py delete mode 100644 requirements.txt delete mode 100644 templates.py delete mode 100644 templates_raw/README.md delete mode 160000 templates_raw/_bundler delete mode 160000 templates_raw/_foreign/normalize.css delete mode 100755 templates_raw/build.sh delete mode 100644 templates_raw/root/.editorconfig delete mode 100644 templates_raw/root/README.md delete mode 100755 templates_raw/root/build.sh delete mode 100644 templates_raw/root/css/feel.css delete mode 100644 templates_raw/root/css/look.css delete mode 100644 templates_raw/root/css/look_bossMode.css delete mode 100644 templates_raw/root/css/look_buttons.css delete mode 100644 templates_raw/root/css/look_hotkeys.css delete mode 100644 templates_raw/root/css/look_stateSwitch.css delete mode 120000 templates_raw/root/css/normalize.css delete mode 100644 templates_raw/root/css/sourceIcons.css delete mode 100644 templates_raw/root/flush delete mode 100644 templates_raw/root/get delete mode 100644 templates_raw/root/js/feel.js delete mode 100644 templates_raw/root/js/feel_hotkeys.js delete mode 100644 templates_raw/root/js/helper.js delete mode 100644 templates_raw/root/js/look_maxSizer.js delete mode 100644 templates_raw/root/js/look_stateSwitch.js delete mode 100644 templates_raw/root/js/snowstorm.js delete mode 100644 templates_raw/root/reset delete mode 100644 templates_raw/root/root.html delete mode 100644 templates_raw/root/test/get_local delete mode 100644 templates_raw/root/test/get_remote delete mode 100644 templates_raw/root/test/testimg.png delete mode 100644 tests/configs/README.md delete mode 100644 tests/configs/factors.ini delete mode 100644 tests/configs/fourchan.ini delete mode 100644 tests/configs/giphy.ini delete mode 100644 tests/configs/instagram.ini delete mode 100644 tests/configs/ninegag.ini delete mode 100644 tests/configs/pr0gramm.ini delete mode 100644 tests/configs/reddit.ini delete mode 100644 tests/configs/soup.ini delete mode 100644 tests/logs/.gitignore diff --git a/.editorconfig b/.editorconfig deleted file mode 100644 index abe2f6ad..00000000 --- a/.editorconfig +++ /dev/null @@ -1,34 +0,0 @@ -root = true - -[*] -end_of_line = LF - -[*.py] -indent_style = space -indent_size = 4 -tab_width = 4 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.css] -indent_style = space -indent_size = 2 -tab_width = 2 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.js] -indent_style = space -indent_size = 2 -tab_width = 2 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.ini] -trim_trailing_whitespace = true -insert_final_newline = true - -[*.md] -indent_style = space -indent_size = 3 -trim_trailing_whitespace = false diff --git a/.gitignore b/.gitignore deleted file mode 100644 index abbacd5a..00000000 --- a/.gitignore +++ /dev/null @@ -1,56 +0,0 @@ - -## ignore our own log files -*.log -logs/* - -## ignore our own custom config -./config.ini - - - -## ignore python runtime files - based on https://github.com/github/gitignore/blob/master/Python.gitignore - -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] - -# C extensions -*.so - -# Distribution / packaging -.Python -env/ -build/ -develop-eggs/ -dist/ -eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -*.egg-info/ -.installed.cfg -*.egg - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.coverage -.cache -nosetests.xml -coverage.xml - -# Translations -*.mo -*.pot - -# Django stuff: -*.log - -# Sphinx documentation -docs/_build/ diff --git a/.gitmodules b/.gitmodules deleted file mode 100644 index 8cbf57b4..00000000 --- a/.gitmodules +++ /dev/null @@ -1,6 +0,0 @@ -[submodule "templates_raw/_bundler"] - path = templates_raw/_bundler - url = https://github.com/jkowalleck/HTML_bundler.git -[submodule "templates_raw/_foreign/normalize.css"] - path = templates_raw/_foreign/normalize.css - url = https://github.com/necolas/normalize.css.git diff --git a/README.md b/README.md index 22bf244a..9b24fc8f 100644 --- a/README.md +++ b/README.md @@ -1,5 +1,14 @@ # nichtparasoup +--- + +:construction: THIS FILE IS OUTDATED +:construction: DUE TO A REWRITE + +--- + + + nichtparasoup is a hackspaces entertainment system. It randomly displays images/gifs from [giphy](https://giphy.com), diff --git a/config.defaults.ini b/config.defaults.ini deleted file mode 100644 index c7883c90..00000000 --- a/config.defaults.ini +++ /dev/null @@ -1,34 +0,0 @@ -;; DEFAULT config for nichtparasoup -;; see README file for more - -;; DO -;; NOT -;; TOUCH - -[General] -Port: 5000 -IP: 0.0.0.0 -Useragent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10) AppleWebKit/600.1.25 (KHTML, like Gecko) Version/8.0 Safari/600.1.25 - -[Cache] -Images_min_limit: 15 - -[Logging] -;; possible destinations: file, syslog -Destination: file -File: nichtparasoup.log -;; verbosity levels: CRITICAL, ERROR, WARNING, INFO, DEBUG, NOTSET -Verbosity: ERROR - -[Sites] -;; no sites default configured. -;; more examples: see the configs dir -;; more examples: see the tests/configs dir -;; just an example: -; SoupIO: everyone -; Pr0gramm: new,top -; Reddit: nsfw,gifs,pics,nsfw_gifs,aww,aww_gifs,reactiongifs,wtf,FoodPorn,cats,StarWars -; NineGag: geeky,wtf,girl,hot,trending -; Instagram: cats,animals,pornhub,nerdy_gaming_art,nature,wtf -; Fourchan: b,sci - diff --git a/configs/demo.ini b/configs/demo.ini deleted file mode 100644 index d53e096d..00000000 --- a/configs/demo.ini +++ /dev/null @@ -1,15 +0,0 @@ -[General] -Port: 5006 - -[Logging] -Destination: syslog -Verbosity: WARNING - -[Sites] -SoupIO: false -Pr0gramm: false -Reddit: gifs*6.0,pics*4.0,aww_gifs,FoodPorn*1.0,cats,EarthPorn -NineGag: false -Instagram: false -Fourchan: false -Giphy: false diff --git a/configs/kadsen.ini b/configs/kadsen.ini deleted file mode 100644 index c0c3bb31..00000000 --- a/configs/kadsen.ini +++ /dev/null @@ -1,9 +0,0 @@ -[General] -Port: 5003 - -[Logging] -Destination: syslog -Verbosity: WARNING - -[Sites] -Reddit: aww_gifs,aww,cats diff --git a/configs/nsfw.ini b/configs/nsfw.ini deleted file mode 100644 index 65aeb57d..00000000 --- a/configs/nsfw.ini +++ /dev/null @@ -1,15 +0,0 @@ -[General] -Port: 5002 - -[Logging] -Destination: Syslog -Verbosity: WARNING - -[Sites] -SoupIO: everyone*5.0 -Pr0gramm: top*9.0,new*0.5 -Reddit: nsfw,gifs,pics,nsfw_gifs,aww_gifs*0.3,reactiongifs,wtf*3.0 -NineGag: false -Instagram: false -Fourchan: b*6.0 -Giphy: false diff --git a/configs/pr0.ini b/configs/pr0.ini deleted file mode 100644 index dd2b7b1a..00000000 --- a/configs/pr0.ini +++ /dev/null @@ -1,9 +0,0 @@ -[General] -Port: 5004 - -[Logging] -Destination: syslog -Verbosity: WARNING - -[Sites] -Pr0gramm: top*9.0,new*0.8 diff --git a/configs/sfw.ini b/configs/sfw.ini deleted file mode 100644 index eb5da651..00000000 --- a/configs/sfw.ini +++ /dev/null @@ -1,14 +0,0 @@ -[General] -Port: 5000 - -[Logging] -Destination: File -File: /tmp/foo.log -Verbosity: debug - -[Sites] -SoupIO: false -Pr0gramm: false -Reddit: gifs*9.0,pics,aww_gifs,mildlyinteresting*3.0,reactiongifs,Oddlysatisfying,Spaceflightporn,Therewasanattempt*5.0,HistoryMemes*3.0,Unexpected,Whatcouldgowrong,Weird,photoshopbattles,NichtDerPostillon,Nonononoyes,earthporn*0.5,Bettereveryloop,assholedesign,machineporn,Shittyfoodporn,Instant_regret,Mildlyinfuriating,somememes,Crappydesign*5.0,Accidentalwesanderson,engineeringporn,hmmm*2.0,mechanical_gifs,cableporn,shittylifehacks*3.0,perfecttiming,AnimalPorn,CatGifs,ChildrenFallingOver,funny,geek,IdiotsInCars,interestingasfuck,itsaunixsystem*5.0,MadeMeSmile,insanepeoplefacebook,PeopleFuckingDying*3.0,spaceflightporn,woahdude*4.0,spaceporn*2.0,neckbeardnests,techsupportgore,perfectloops,wasletztepreis -Instagram: false -Fourchan: false diff --git a/crawler/__init__.py b/crawler/__init__.py deleted file mode 100644 index 76411ff9..00000000 --- a/crawler/__init__.py +++ /dev/null @@ -1,442 +0,0 @@ - -__all__ = ['Crawler', 'CrawlerError', 'ImageData'] - - -import sys -import random -import time -import re - -try: - from urllib.request import Request, urlopen # py3 -except ImportError: - from urllib2 import Request, urlopen # py2 - -try: - from urllib.request import URLError as HTTPError # py3 -except ImportError: - from urllib2 import HTTPError # py2 - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - -from bs4 import BeautifulSoup - - -class Crawler(object): - """ - abstract class Crawler - """ - - ## class constants - - __C_timeout_ = 'timeout' - __C_headers_ = 'headers' - __C_resetDelay_ = 'resetDelay' - - __refresh_uri_RE = re.compile(r'^(?:\d*;)?url=(.+)$', flags=re.IGNORECASE) - - __image_RE = re.compile(r'.*\.(?:jpeg|jpg|png|gif)(?:\?.*)?(?:#.*)?$', flags=re.IGNORECASE) - - ## class vars - - __configuration = { - __C_timeout_: 10, # needs to be greater 0 - __C_headers_: {}, # additional headers - __C_resetDelay_: 10800 # value in seconds - } - - __blacklist = [] - __images = {} - - __factors = None - __logger = None - - ## properties - - __crawlingStarted = 0.0 - - ## config methods - - @classmethod - def configure(cls, config=None, key=None, value=None): - if isinstance(config, dict): - cls.__configuration += config - elif key and value: - cls.__configuration[key] = value - - @classmethod - def __config_setter_and_getter(cls, key, value=None): - if value is None: - return cls.__configuration[key] - cls.configure(key=key, value=value) - - ## wellknown config accessors - - @classmethod - def request_headers(cls, value=None): - return cls.__config_setter_and_getter(cls.__C_headers_, value) - - @classmethod - def request_timeout(cls, value=None): - return cls.__config_setter_and_getter(cls.__C_timeout_, value) - - @classmethod - def reset_delay(cls, value=None): - return cls.__config_setter_and_getter(cls.__C_resetDelay_, value) - - ## basic document fetcher - - @classmethod - def __html_find_meta_refresh(cls, document): - """ - :type document: BeautifulSoup - :rtype: str | None - """ - refresh_uri = None - - # - meta_refresh = document.find("meta", {"http-equiv": "refresh", "content": cls.__refresh_uri_RE}) - if meta_refresh: - # @fixme html-decode ! - refresh_uri = cls.__refresh_uri_RE.match(meta_refresh["content"]).group(1) - - return refresh_uri - - @staticmethod - def __html_find_base(document): - """ - :type document: BeautifulSoup - :rtype: str | None - """ - base_uri = None - - # - base_tag = document.find("base", {"href": True}) - if base_tag: - # @fixme html-decode ! - base_uri = base_tag["href"] - - return base_uri - - @classmethod - def _fetch_remote(cls, uri, depth_indicator=1): - """ - return remote document and actual remote uri - :type uri: str - :type depth_indicator: int - :rtype: (document: str | None, uri) - """ - - cls._log("debug", "fetch remote(%d): %s" % (depth_indicator, uri)) - request = Request(uri, headers=cls.request_headers()) - try: - response = urlopen(request, timeout=cls.request_timeout()) - except HTTPError: - cls._log("warn", "Could not fetch: %s" % uri) - return None - - if not response: - return None - - uri = response.geturl() - - charset = 'utf8' - try: - charset = response.info().get_param('charset', charset) # py3 - except AttributeError: - pass - - return response.read().decode(charset), uri - - @classmethod - def _fetch_remote_html(cls, uri, follow_meta_refresh=True, follow_meta_refresh_max=5, bs4features="html5lib"): - """ - returns remote HTML document, actual remote uri and base uri - :type uri: str - :type follow_meta_refresh: bool - :type follow_meta_refresh_max: int - :rtype: ( document: BeautifulSoup | None , base: str, uri: str ) - """ - - document = None - follow_meta_refresh_depth = 1 - - while True: - (response, uri) = cls._fetch_remote(uri, follow_meta_refresh_depth) - - if not response: - break - - document = BeautifulSoup(response, features=bs4features) - - if not follow_meta_refresh: - break - - refresh_uri = cls.__html_find_meta_refresh(document) - if not refresh_uri: - break - refresh_uri = urljoin(uri, refresh_uri) - - if refresh_uri == uri: - break - - cls._log("debug", "fetch remote HTML(%d): %s meta-refreshes to %s" % - (follow_meta_refresh_depth, uri, refresh_uri)) - uri = refresh_uri - - follow_meta_refresh_depth += 1 - if follow_meta_refresh_depth >= follow_meta_refresh_max: - break - - if not document: - cls._log("debug", "fetch remote HTML: %s is empty" % uri) - - base = uri - - doc_base = cls.__html_find_base(document) - if doc_base: - base = urljoin(base, doc_base) - - return document, base, uri - - ## general functions - - @classmethod - def __blacklist_clear(cls): - cls.__blacklist = [] # be aware: list.clean() is not available in py2 - - @classmethod - def _blacklist(cls, uri): - cls.__blacklist.append(uri) - - @classmethod - def _is_blacklisted(cls, uri): - return uri in cls.__blacklist - - @classmethod - def _is_image(cls, uri): - return cls.__image_RE.match(uri) is not None - - @classmethod - def __images_clear(cls): - cls.__images = {} # be aware: dict.clean() is not available in py2 - - @classmethod - def __add_image(cls, uri, crawler, site, source, more): - """ - :type uri: str - :type crawler: str - :type site: str - :type source: str|None - :type more: dict - :rtype: bool - """ - if not cls._is_image(uri): - # self._log("info", uri + " is no image ") - return False - - if cls._is_blacklisted(uri): - return False - - if crawler not in cls.__images: - cls.__images[crawler] = {} - - if site not in cls.__images[crawler]: - cls.__images[crawler][site] = [] - - cls._blacklist(uri) # add it to the blacklist to detect duplicates - cls.__images[crawler][site].append(ImageData(uri, source, more)) - cls._log("debug", "added %s-%s-%s: %s" % (crawler, site, source, uri)) - return True - - @classmethod - def get_image(cls): - """ - :rtype: dict - """ - images = cls.__images - factors = cls.__factors - if images: - # return image respecting factors: - # Summing up the Factors till reaching a random number - - max = 0 - for crawler in images: - for site in images[crawler]: - factor = 1 - if crawler in factors and site in factors[crawler]: - factor = factors[crawler][site] - - max += factor - - rnd = random.uniform(0, max) - - count = 0 - for crawler in images: - for site in images[crawler]: - factor = 1 - if crawler in factors and site in factors[crawler]: - factor = factors[crawler][site] - count += factor - if count <= rnd: - continue - - image = random.choice(images[crawler][site]) - if not image: - continue - - images[crawler][site].remove(image) - cls._log("debug", "delivered %s-%s: %s - remaining: %d" - % (crawler, site, image.uri, len(images[crawler][site]))) - - result = image.asDict() - result['crawler'] = crawler - result['site'] = site - return result - return None - - @classmethod - def set_logger(cls, logger): - cls.__logger = logger - - @classmethod - def set_factors(cls, factors): - cls.__factors = factors - - @classmethod - def _log(cls, log_type, message): - if cls.__logger: - getattr(cls.__logger, log_type)(message) - - @classmethod - def _debug(cls): - return "" % (cls.__configuration, cls.info()) - - @classmethod - def info(cls): - images = cls.__images - blacklist = cls.__blacklist - - info = { - "blacklist": len(blacklist), - "blacklist_size": sys.getsizeof(blacklist, 0), - "images_per_site": {} - } - - images_count = 0 - images_size = 0 - for crawler in images: - for site in images[crawler]: - site_image_count = len(images[crawler][site]) - images_count += site_image_count - images_size += sys.getsizeof(images[crawler][site]) - info["images_per_site"][crawler + "_" + site] = site_image_count - - info["images_size"] = images_size - info["images"] = images_count - - return info - - @classmethod - def show_imagelist(cls): - imagelist = [] - for crawler in cls.__images: - for site in cls.__images[crawler]: - for imageData in cls.__images[crawler][site]: - imagelist.append(imageData.uri) - return imagelist - - @classmethod - def show_blacklist(cls): - return cls.__blacklist - - @classmethod - def flush(cls): - cls.__images_clear() - cls._log("info", "flushed. cache is now empty") - - @classmethod - def reset(cls): - cls.__images_clear() - cls.__blacklist_clear() - cls._log("info", "reset. cache and blacklist are now empty") - - def crawl(self): - now = time.time() - if not self.__crawlingStarted: - self.__crawlingStarted = now - elif self.__crawlingStarted <= now - self.__class__.reset_delay(): - self.__class__._log("debug", "instance %s starts at front" % (repr(self))) - self.__crawlingStarted = now - self._restart_at_front() - - self.__class__._log("debug", "instance %s starts crawling" % (repr(self))) - try: - self._crawl() - except TypeError: - # Catch empty returns from crawling (returns nonetype) - pass - except CrawlerError as e: - self.__class__._log("exception", "crawler error: %s" % (repr(e))) - except: - e = sys.exc_info()[0] - self.__class__._log("exception", "unexpected crawler error: %s" % (repr(e))) - - def _add_image(self, uri, site, source=None, **kwargs): - """ - :type uri: str - :type site: str - :type source: str|None - :type kwargs: dict - :rtype: bool - """ - return self.__class__.__add_image(uri, self.__class__.__name__, site, source, kwargs) - - ## abstract functions - - def __init__(self): - # call every abstract method to be sure it is there - self._restart_at_front() - self._crawl() - raise NotImplementedError("Should have implemented this") - - def _crawl(self): - raise NotImplementedError("Should have implemented this") - - def _restart_at_front(self): - raise NotImplementedError("Should have implemented this") - - -class CrawlerError(Exception): - pass - - -class ImageData(object): - """ class def: image data """ - - uri = "" - source = None - more = {} - - def __init__(self, uri, source=None, more={}): - """ - :type uri: str - :type source: str|None - :type more: dict - """ - self.uri = uri - self.source = source - self.more = more - - def asDict(self): - """ - :rtype: dict - """ - return { - "uri": self.uri , - "source": self.source , - "more": self.more , - } diff --git a/crawler/fourchan.py b/crawler/fourchan.py deleted file mode 100644 index e5818eed..00000000 --- a/crawler/fourchan.py +++ /dev/null @@ -1,59 +0,0 @@ - - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - - -from . import Crawler, CrawlerError - - -class Fourchan(Crawler): - """ Fourchan image provider """ - - __uri = "" - __next = "" - __site = "" - - @staticmethod - def __build_uri(uri): - return uri - - def _restart_at_front(self): - self.__next = self.__uri - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__.__build_uri(uri) - self._restart_at_front() - - def _crawl(self): - uri = self.__next - if not uri: - return - - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - (page, base, _) = self.__class__._fetch_remote_html(uri) - if not page: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - # get more content ("scroll down") - # to know what page to parse next - # update new last URI when we're not on first run - _buttons = page.find_all("a", {"class": "button", "href": True}) - if _buttons: - self.__next = urljoin(base, _buttons[-1]["href"]) - else: - self.__class__._log("debug", "%s found no `next` on url: %s" % (self.__class__.__name__, uri)) - self.__next = "" - - images_added = 0 - for con in page.find_all("a", {"class": "fileThumb", "href": True}): - if self._add_image(urljoin(base, con["href"]), self.__site): - images_added += 1 - - if not images_added: - self.__class__._log("debug", "%s found no images on url: %s" % (self.__class__.__name__, uri)) diff --git a/crawler/giphy.py b/crawler/giphy.py deleted file mode 100644 index 5c0b9c59..00000000 --- a/crawler/giphy.py +++ /dev/null @@ -1,52 +0,0 @@ - - -import json - -from . import Crawler, CrawlerError - - -class Giphy(Crawler): - """ class def: a crawler for Giphy """ - - __uri = "" - __next = 0 - __site = "" - - __limit = 50 - - __api_key = "dc6zaTOxFJmzC" - - @classmethod - def _build_uri(cls, uri): - return uri + "&api_key=" + cls.__api_key + "&limit=" + str(cls.__limit) - - def _restart_at_front(self): - self.__next = 0 - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__._build_uri(uri) - self._restart_at_front() - - def _crawl(self): - uri = self.__uri + "&offset=" + str(self.__next) - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - (remote, uri) = self.__class__._fetch_remote(uri) - if not remote: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - data = json.loads(remote) - - self.__next += self.__limit - - images_added = 0 - for child in data['data']: - image = child['images']['original']['url'] - if image: - if self._add_image(image, self.__site): - images_added += 1 - - if not images_added: - self.__class__._log("debug", "%s found no images on url: %s" % (self.__class__.__name__, uri)) diff --git a/crawler/instagram.py b/crawler/instagram.py deleted file mode 100644 index e5454883..00000000 --- a/crawler/instagram.py +++ /dev/null @@ -1,65 +0,0 @@ - - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - -import json - - -from . import Crawler, CrawlerError - - -class Instagram(Crawler): - """ instagram image provider """ - - __uri = "" - __last = "" - __site = "" - - ## class methods - - @staticmethod - def __build_uri(uri): - return urljoin(uri+"/", "./media/") - - ## instance methods - - def _restart_at_front(self): - self.__last = "" - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__.__build_uri(uri) - self._restart_at_front() - - def _crawl(self): - uri = urljoin(self.__uri, "?max_id="+self.__last) - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - (remote, uri) = self.__class__._fetch_remote(uri) - if not remote: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - data = json.loads(remote) - if "status" not in data or data["status"] != "ok": - raise CrawlerError() - - images_added = 0 - for item_image in [item for item in data['items'] if item["type"] == "image"]: - if 'id' in item_image: - self.__last = item_image['id'] - - if 'images' in item_image \ - and 'standard_resolution' in item_image['images']\ - and 'url' in item_image['images']['standard_resolution']: - image = item_image['images']['standard_resolution']['url'] - if image: - if self._add_image(image, self.__site): - images_added += 1 - - if not images_added: - self.__class__._log("debug", "%s found no images on url: %s" % (self.__class__.__name__, uri)) - diff --git a/crawler/ninegag.py b/crawler/ninegag.py deleted file mode 100644 index 32033e5f..00000000 --- a/crawler/ninegag.py +++ /dev/null @@ -1,78 +0,0 @@ - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - -import re - - -from . import Crawler, CrawlerError - - - -class NineGag(Crawler): - """ 9gag image provider """ - - __uri = "" - __next = "" - __site = "" - - __RE_post_container = re.compile(r'badge-post-container') - - @staticmethod - def __build_uri(uri): - return uri - - def _restart_at_front(self): - self.__next = self.__uri - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__.__build_uri(uri) - self._restart_at_front() - - def _crawl(self): - uri = self.__next - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - (page, base, _) = self.__class__._fetch_remote_html(uri) - if not page: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - # get more content ("scroll down") - # to know what page to parse next - # update new last URI when we're not on first run - _more = page.find("div", {"class": "loading"}) - _next = None - if _more: - _more = _more.find("a", {"class": "btn badge-load-more-post", "href": True}) - if _more: - _next = urljoin(base, _more["href"]) - if _next: - self.__next = _next - else: - self.__class__._log("debug", "%s found no `next` on url: %s" % (self.__class__.__name__, uri)) - - - # for every found imageContainer - # add img-src to map if not blacklisted - images_added = 0 - for con in page.find_all("div", {"class": self.__class__.__RE_post_container}): - image = None - - image_inline = con.find("div", {"class": "badge-animated-container-animated", "data-image": True}) - if image_inline: - image = image_inline['data-image'] - - image_src = con.find('img', {"class": "badge-item-img", "src": True}) - if image_src: - image = image_src['src'] - - if image: - if self._add_image(urljoin(base, image), self.__site): - images_added += 1 - - if not images_added: - self.__class__._log("debug", "%s found no images on url: %s" % (self.__class__.__name__, uri)) diff --git a/crawler/pr0gramm.py b/crawler/pr0gramm.py deleted file mode 100644 index 03387588..00000000 --- a/crawler/pr0gramm.py +++ /dev/null @@ -1,105 +0,0 @@ - -import re, json - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - -from . import Crawler, CrawlerError - - -class Pr0gramm(Crawler): - """ pr0gramm.com image provider""" - - ## class constants - - ## properties - - __uri = "" - __next = "" - __site = "" - __image_base_url = "https://img.pr0gramm.com/" - __api_base_url = "" - - __filter = re.compile(r'^/static/[\d]+') - __filterNextPage = "" - - ## functions - - """ - JSON From API - { - atEnd: - atStart: - error: ? - items: [ - ... - { - id: - promoted: - up: - down: - created: - image: z.B.: 2016/02/18/047b2e356d059074.jpg - thumb: - fullsize: - source: - flags: - user: - mark: - } - ... - ] - ts: - cache: - rt: - qc: - } - """ - - @staticmethod - def __build_uri(uri): - return uri - - def _restart_at_front(self): - self.__next = self.__uri - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__.__build_uri(uri) - self.__api_base_url = self.__uri - self.__filterNextPage = re.compile(r'^/static/' + re.escape(self.__site) + r'/[\d]+') - self._restart_at_front() - - def _crawl(self): - uri = self.__next - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - - (remote, uri) = self.__class__._fetch_remote(uri) - if not remote: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - data = json.loads(remote) - - last_id = 0 - images_added = 0 - for item in data["items"]: - - if item["promoted"] == 0: - continue - - if item["image"].lower().endswith(".webm"): - continue - - last_id = item["id"] - - if self._add_image(urljoin(self.__image_base_url, item["image"]), self.__site): - images_added += 1 - - - self.__next = self.__api_base_url + "?older=" + str(last_id) - - self.__class__._log("debug", "%s added %s images on url: %s" % (self.__class__.__name__, images_added, uri)) diff --git a/crawler/reddit.py b/crawler/reddit.py deleted file mode 100644 index 596495f6..00000000 --- a/crawler/reddit.py +++ /dev/null @@ -1,55 +0,0 @@ - - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - -import json - -from . import Crawler, CrawlerError - - -class Reddit(Crawler): - """ class def: a crawler for Reddit image threads """ - - __uri = "" - __next = "" - __site = "" - - @staticmethod - def __build_uri(uri): - return uri + ".json" - - def _restart_at_front(self): - self.__next = "" - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__.__build_uri(uri) - self._restart_at_front() - - def _crawl(self): - uri = urljoin(self.__uri, "?after="+self.__next) - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - (remote, uri) = self.__class__._fetch_remote(uri) - if not remote: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - data = json.loads(remote) - - self.__next = data['data']['after'] - - images_added = 0 - for child in data['data']['children']: - image = child['data']['url'] - if image: - thread_uri = urljoin(self.__uri, child['data']['permalink']) - self.__class__._log("debug", thread_uri) - if self._add_image(image, self.__site, thread_uri): - images_added += 1 - - if not images_added: - self.__class__._log("debug", "%s found no images on url: %s" % (self.__class__.__name__, uri)) diff --git a/crawler/soupio.py b/crawler/soupio.py deleted file mode 100644 index 2d15e2eb..00000000 --- a/crawler/soupio.py +++ /dev/null @@ -1,63 +0,0 @@ - - -try: - from urllib.parse import urljoin # py3 -except ImportError: - from urlparse import urljoin # py2 - -from . import Crawler, CrawlerError - - -class SoupIO(Crawler): - """ soup.io image provider """ - - __uri = "" - __next = "" - __site = "" - - @staticmethod - def __build_uri(uri): - return urljoin(uri, "?type=image") - - def _restart_at_front(self): - self.__next = self.__uri - - def __init__(self, uri, site): - self.__site = site - self.__uri = self.__class__.__build_uri(uri) - self._restart_at_front() - - def _crawl(self): - uri = urljoin(self.__uri, self.__next) - self.__class__._log("debug", "%s crawls url: %s" % (self.__class__.__name__, uri)) - - (page, base, _) = self.__class__._fetch_remote_html(uri) - if not page: - self.__class__._log("debug", "%s crawled EMPTY url: %s" % (self.__class__.__name__, uri)) - return - - # get more content ("scroll down") - # to know what page to parse next - # update new last URI when we're not on first run - _next = None - _more = page.find("div", {"id": "more_loading"}) - if _more: - _more = _more.find("a", {"href": True}) - if _more: - _next = urljoin(base, _more["href"]) - if _next: - self.__next = _next - else: - self.__class__._log("debug", "%s found no `next` on url: %s" % (self.__class__.__name__, uri)) - - # for every found imageContainer - # add img-src to map if not blacklisted - images_added = 0 - for con in page.find_all("div", {"class": "imagecontainer"}): - image = con.find('img', {"src": True}) - if image: - if self._add_image(urljoin(base, image['src']), self.__site): - images_added += 1 - - if not images_added: - self.__class__._log("debug", "%s found no images on url: %s" % (self.__class__.__name__, uri)) diff --git a/nichtparasoup.py b/nichtparasoup.py deleted file mode 100755 index 39d06804..00000000 --- a/nichtparasoup.py +++ /dev/null @@ -1,402 +0,0 @@ -#!/usr/bin/env python - -### import libraries -from crawler.giphy import Giphy -from crawler.fourchan import Fourchan -from crawler.instagram import Instagram -from crawler.ninegag import NineGag -from crawler.pr0gramm import Pr0gramm -from crawler.soupio import SoupIO -from crawler.reddit import Reddit -from os import path -import math -import random -import logging -import logging.handlers -import time -import threading -import argparse -import json - -try: - from configparser import RawConfigParser # py3 -except ImportError: - from ConfigParser import RawConfigParser # py2 - -try: - from urllib.parse import quote_plus as url_quote_plus # py3 -except ImportError: - from urllib import quote_plus as url_quote_plus # py2 - -from werkzeug.wrappers import Request, Response -from werkzeug.routing import Map, Rule -from werkzeug.exceptions import HTTPException, NotFound -from werkzeug.serving import run_simple - - -## import templates -import templates as tmpl - -## import crawler -from crawler import Crawler - -_file_path = path.dirname(path.realpath(__file__)) - -# argument parser -arg_parser = argparse.ArgumentParser() -arg_parser.add_argument('-c', '--config-file', metavar='', - type=argparse.FileType('r'), - required=True, - help='a file path to the config ini', - dest="config_file") -args = arg_parser.parse_args() - -# configuration -# init config parser -config = RawConfigParser() - -# read defaults -config.read(path.join(_file_path, 'config.defaults.ini')) -try: - config.read_file(args.config_file) # py3 -except AttributeError: - config.readfp(args.config_file) # py2 -args.config_file.close() - -# get actual config items -nps_port = config.getint("General", "Port") -nps_bindip = config.get("General", "IP") - -min_cache_imgs_before_refill = config.getint("Cache", "Images_min_limit") -user_agent = config.get("General", "Useragent") - -# crawler logging config -logverbosity = config.get("Logging", "Verbosity") -logger = logging.getLogger("nichtparasoup") - -if config.get("Logging", "Destination").lower() == 'syslog': - hdlr = logging.handlers.SysLogHandler() -else: - hdlr = logging.FileHandler(config.get("Logging", "File")) - hdlr.setFormatter(logging.Formatter( - '%(asctime)s %(levelname)s %(message)s')) - -logger.addHandler(hdlr) -logger.setLevel(logverbosity.upper()) - -# werkzeug logging config -log = logging.getLogger('werkzeug') -log.setLevel(logging.CRITICAL) - -try: - urlpath = config.get("General", "Urlpath") -except: - urlpath = '' - -factor_separator = "*" -call_flush_timeout = 10 # value in seconds -call_flush_last = time.time() - call_flush_timeout - -call_reset_timeout = 10 # value in seconds -call_reset_last = time.time() - call_reset_timeout - -Crawler.request_headers({'User-Agent': user_agent}) -Crawler.set_logger(logger) - -# config the crawlers - - -def get_crawlers(configuration, section): - """ - parse the config section for crawlers - * does recognize (by name) known and implemented crawlers only - * a robust config reading and more freedom for users - - :param configuration: RawConfigParser - :param section: string - :return: crawler, factors - """ - crawlers = {} - factors = {} - - for crawler_class in Crawler.__subclasses__(): - crawler_class_name = crawler_class.__name__ - if not configuration.has_option(section, crawler_class_name): - continue # skip crawler if not configured - - crawler_config = configuration.get(section, crawler_class_name) - if not crawler_config or crawler_config.lower() == "false": - continue # skip crawler if not configured or disabled - - crawler_uris = {} - - # mimic old behaviours for bool values - if crawler_config.lower() == "true": - if crawler_class == SoupIO: - crawler_config = "everyone" - - crawler_sites_and_factors = [site_stripped for site_stripped in [site.strip() for site in crawler_config.split(",")] # trim sites - if site_stripped] # filter stripped list for valid values - - if not crawler_sites_and_factors: - continue # skip crawler if no valid sites configured - - crawler_sites = [] - factors[crawler_class_name] = {} - - # Separate Site and Factor - for factorPair in crawler_sites_and_factors: - if factor_separator not in factorPair: - # No Factor configured - crawler_sites.append(url_quote_plus(factorPair)) - continue - - factorPair_parts = [factorPairPart.strip( - ) for factorPairPart in factorPair.split(factor_separator)] - - if not factorPair_parts or not len(factorPair_parts) == 2: - continue - - site = url_quote_plus(factorPair_parts[0]) - factor = float(factorPair_parts[1]) - - crawler_sites.append(site) - - if site not in factors[crawler_class_name] and 0 < factor <= 10: - factors[crawler_class_name][site] = factor - - logger.info("found configured Crawler: %s = %s Factors: %s" % ( - crawler_class_name, repr(crawler_sites), repr(factors[crawler_class_name]))) - - if crawler_class == Reddit: - crawler_uris = {site: "https://www.reddit.com/r/%s" % - site for site in crawler_sites} - elif crawler_class == NineGag: - crawler_uris = {site: "https://9gag.com/%s" % - site for site in crawler_sites} - elif crawler_class == Pr0gramm: - crawler_uris = {crawler_sites[0] : "https://pr0gramm.com/api/items/get"} - elif crawler_class == SoupIO: - crawler_uris = {site: ("http://www.soup.io/%s" if site in ["everyone"] # public site - else "http://%s.soup.io") % site # user site - for site in crawler_sites} - elif crawler_class == Instagram: - crawler_uris = {site: "https://instagram.com/%s" % - site for site in crawler_sites} - elif crawler_class == Fourchan: - crawler_uris = {site: "https://boards.4chan.org/%s/" % - site for site in crawler_sites} - elif crawler_class == Giphy: - crawler_uris = { - site: "https://api.giphy.com/v1/gifs/search?q=%s" % site for site in crawler_sites} - - if crawler_class_name not in crawlers: - crawlers[crawler_class_name] = {} - - crawlers[crawler_class_name] = {site: crawler_class( - crawler_uris[site], site) for site in crawler_uris} - - return crawlers, factors - - -(sources, factors) = get_crawlers(config, "Sites") -if not sources: - raise Exception("no sources configured") -if factors: - Crawler.set_factors(factors) - - -# wrapper function for cache filling -def cache_fill_loop(): - global sources - while True: # fill cache up to min_cache_imgs per site - - info = Crawler.info() - for crawler in sources: - for site in sources[crawler]: - key = crawler + "_" + site - - if key not in info["images_per_site"] or info["images_per_site"][key] < min_cache_imgs_before_refill: - try: - sources[crawler][site].crawl() - info = Crawler.info() - except Exception as e: - logger.error("Error in crawler %s - %s: %s" % - (crawler, site, e)) - break - - # sleep for non-invasive threading ;) - time.sleep(1.337) - - -# return image data from map -def cache_get(): - return Crawler.get_image() - -# get status of cache - - -def cache_status_dict(): - info = Crawler.info() - return { - "crawler": Crawler.info(), - "factors": factors, - "min_cache_imgs_before_refill": min_cache_imgs_before_refill, - } - -# print status of cache - - -def cache_status_text(): - status = cache_status_dict() - info = status['crawler'] - - bar_reps = 5 - bar_repr_refill = status['min_cache_imgs_before_refill'] / bar_reps - - msg = "images cached: %d (%d bytes) - already crawled: %d (%d bytes)" % \ - (info["images"], info["images_size"], - info["blacklist"], info["blacklist_size"]) - logger.info(msg) - - for crawler in sources: - for site in sources[crawler]: - key = crawler + "_" + site - if key in info["images_per_site"]: - - factor = 1 - if crawler in factors and site in factors[crawler]: - factor = factors[crawler][site] - - count = info["images_per_site"][key] - - bar = "|" - for i in range(0, int(math.ceil(count / bar_reps))): - if i < bar_repr_refill: - bar += "#" - else: - bar += "*" - - sitestats = ("%15s - %-15s with factor %4.1f: %2d Images " + - bar) % (crawler, site, factor, count) - logger.info(sitestats) - msg += "\r\n" + sitestats - return msg - -# print imagelist - - -def show_imagelist(): - return "\n".join(Crawler.show_imagelist()) - - -# print blacklist -def show_blacklist(): - return "\n".join(Crawler.show_blacklist()) - - -# flush blacklist -def flush(): - global call_flush_last - time_since_last_call = time.time() - call_flush_last - if time_since_last_call >= call_flush_timeout: - Crawler.flush() - call_flush_last = time.time() - time_since_last_call = 0 - return "%i000" % (call_flush_timeout - time_since_last_call) - - -# reset the crawler -def reset(): - global call_reset_last - time_since_last_call = time.time() - call_reset_last - if time_since_last_call >= call_reset_timeout: - Crawler.reset() - call_reset_last = time.time() - time_since_last_call = 0 - return "%i000" % (call_reset_timeout - time_since_last_call) - - -# werkzeug webserver -# class with mapping to cache_* functions above -class NichtParasoup(object): - # init webserver with routing - def __init__(self): - self.url_map = Map([ - Rule(urlpath + '/', endpoint='root'), - Rule(urlpath + '/status', endpoint='cache_status'), - Rule(urlpath + '/get', endpoint='cache_get'), - Rule(urlpath + '/imagelist', endpoint='show_imagelist'), - Rule(urlpath + '/blacklist', endpoint='show_blacklist'), - Rule(urlpath + '/flush', endpoint='flush'), - Rule(urlpath + '/reset', endpoint='reset'), - ]) - - # proxy call to the wsgi_app - def __call__(self, environ, start_response): - return self.wsgi_app(environ, start_response) - - # calculate the request and use the defined map to route - def dispatch_request(self, request): - adapter = self.url_map.bind_to_environ(request.environ) - try: - endpoint, values = adapter.match() - return getattr(self, 'on_' + endpoint)(request, **values) - except HTTPException as e: - return e - - # the wsgi app itself - def wsgi_app(self, environ, start_response): - request = Request(environ) - response = self.dispatch_request(request) - return response(environ, start_response) - - # start page with js and scroll - def on_root(self, request): - return Response(tmpl.root, mimetype='text/html') - - # map function for print the status - def on_cache_status(self, request): - if request.values.get('t') == 'json': - return Response(json.dumps(cache_status_dict()), mimetype='application/json') - return Response(cache_status_text()) - - # map function for getting an image url - def on_cache_get(self, request): - return Response(json.dumps(cache_get()), mimetype='application/json') - - # map function for showing blacklist - def on_show_blacklist(self, request): - return Response(show_blacklist()) - - # map function for showing imagelist - def on_show_imagelist(self, request): - return Response(show_imagelist()) - - # map function for flushing (deleting everything in cache) - def on_flush(self, request): - return Response(flush()) - - # map function for resetting (deleting everything in cache and blacklist) - def on_reset(self, request): - return Response(reset()) - - -# runtime -# main function how to run -# on start-up, fill the cache and get up the webserver -if __name__ == "__main__": - try: - # start the cache filler tread - cache_fill_thread = threading.Thread(target=cache_fill_loop) - cache_fill_thread.daemon = True - cache_fill_thread.start() - except (KeyboardInterrupt, SystemExit): - # end the cache filler thread properly - min_cache_imgs_before_refill - 1 - - # give the cache_fill some time in advance - time.sleep(1.337) - - # start webserver after a bit of delay - run_simple(nps_bindip, nps_port, NichtParasoup(), use_debugger=False) diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index 6b7496c4..00000000 --- a/requirements.txt +++ /dev/null @@ -1,9 +0,0 @@ -### the webserver toolkit -Werkzeug - -### basic HTML/XML toolkit -beautifulsoup4 - -### advanced html/xml parser & tree builders -html5lib -lxml diff --git a/templates.py b/templates.py deleted file mode 100644 index ca49d09e..00000000 --- a/templates.py +++ /dev/null @@ -1,374 +0,0 @@ - -root = """ - -nichtparasoup - Hackspace Entertainment System - - - - -
- -  - -
- 
esc
boss mode
-
space
play/pause
+
faster -
-
slower
j
next image
-
k
next-to-last image
hot keys
-""" diff --git a/templates_raw/README.md b/templates_raw/README.md deleted file mode 100644 index 0dc89ebe..00000000 --- a/templates_raw/README.md +++ /dev/null @@ -1,26 +0,0 @@ -# template builder for UI - -This is just the template builder - not needed to run __nichtparasoup__. - -## idea - -To have proper IDE support when it comes to writing the UI, the thing should be written in several raw files and be -yuiCompressed and put together at the end. -Furthermore it would be great to be able to develop the UI without running __nichtparasoup__. - -## requirements - -```sh -git submodule update --init --recursive -``` - -for further information: see the `README` of the submodules - -## how to use - -just run the `build.sh` so the `templates.py` of __nichtparasoup__ will be built automatically - -## attention: symlinks - -some files are symlinked - so be sure your environment supports symlinks, otherwise the outcome may be broken. -this is the case on most windows systems. diff --git a/templates_raw/_bundler b/templates_raw/_bundler deleted file mode 160000 index 5acc19fa..00000000 --- a/templates_raw/_bundler +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 5acc19fa3c2803be416a7941acb6f9fa65aa81ee diff --git a/templates_raw/_foreign/normalize.css b/templates_raw/_foreign/normalize.css deleted file mode 160000 index d8313280..00000000 --- a/templates_raw/_foreign/normalize.css +++ /dev/null @@ -1 +0,0 @@ -Subproject commit d83132802e8bc10ece0c0afbe88fce83674ec252 diff --git a/templates_raw/build.sh b/templates_raw/build.sh deleted file mode 100755 index aabd2f2c..00000000 --- a/templates_raw/build.sh +++ /dev/null @@ -1,34 +0,0 @@ -#!/bin/sh - - -oldDir=$(pwd) - -cd $(dirname $0) - - -###################### - -target="../templates.py" - -###################### - -printf '\n' > $target - -for builder in */build.sh -do - echo "running $builder ... " - - printf $(basename $(dirname $builder)) >> $target - printf ' = """' >> $target - - $builder 1>> $target - - printf '"""' >> $target - -done - -printf '\n' >> $target - -##################### - -cd $oldDir diff --git a/templates_raw/root/.editorconfig b/templates_raw/root/.editorconfig deleted file mode 100644 index f059efd6..00000000 --- a/templates_raw/root/.editorconfig +++ /dev/null @@ -1,36 +0,0 @@ -root = true - -[*] -end_of_line = LF - -[flush,reset,get,get_local,get_remote] -end_of_line = LF -trim_trailing_whitespace = false -insert_final_newline = false - -[*.html] -indent_style = space -indent_size = 2 -tab_width = 2 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.css] -indent_style = tab -indent_size = 2 -tab_width = 2 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.js] -indent_style = tab -indent_size = 2 -tab_width = 2 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.md] -indent_style = space -indent_size = 3 -trim_trailing_whitespace = false -insert_final_newline = true diff --git a/templates_raw/root/README.md b/templates_raw/root/README.md deleted file mode 100644 index b0e24219..00000000 --- a/templates_raw/root/README.md +++ /dev/null @@ -1,26 +0,0 @@ -# ROOT LAYOUT dev env - -~ this is a local runnable UI builder environment that needs no web server and no internet connection (just `file://`). -Ideal for development on long train rides ;-) - -~ since this development may require XHttpRequests (= AJAX) you want to use a web browser that supports such requests via `file://` - like FireFox - or start a local web server via -`python -m SimpleHTTPServer `. - -## how to develop - -The file `root.html` is the main UI file for _nichtparasoup_. -It may include any known HTML, JS and CSS you need. - -The UI file(s) will be bundled. For details see the `build.sh`. -Therefore some specialties are needed to know: - -* the HTML tag `striponbuild` will be removed on build and may be used for debug/develop purposes -* each line of CSS or JS that contains the string `@striponbuild` is stripped on build, so this may be used for development purposes also - -/!\ attention: -the `normalize.css` is symlinked. symlinks may not work properly on windows systems. - -## how to test - -The ready made UI will normally do a AJAX request to `./get` which should return the URI of an image. -For testing purposes and local dev, a file `test/get_local` was prepared which will respond a locally stored image. Since this Image will have the same Size every time, there is also a file `test/get_remote` which used to respond a random (resolution/content) image. I guess you know how to handle them. diff --git a/templates_raw/root/build.sh b/templates_raw/root/build.sh deleted file mode 100755 index bd858395..00000000 --- a/templates_raw/root/build.sh +++ /dev/null @@ -1,10 +0,0 @@ -#!/bin/sh - -dir=$(dirname $0) - -bundler="$dir/../_bundler//bundler.py --compress --strip-comments \ ---strip-tags striponbuild \ ---strip-markers @striponbuild" - -$bundler -i $dir/root.html - diff --git a/templates_raw/root/css/feel.css b/templates_raw/root/css/feel.css deleted file mode 100644 index ec21b5e0..00000000 --- a/templates_raw/root/css/feel.css +++ /dev/null @@ -1,11 +0,0 @@ -/* remember for development: lines that include the string "@stripOnBuild" will be stripped on build ;-) */ - -input[type="range"] { - position: relative; - top: 5px; - direction: rtl; -} - - - - diff --git a/templates_raw/root/css/look.css b/templates_raw/root/css/look.css deleted file mode 100644 index b612160b..00000000 --- a/templates_raw/root/css/look.css +++ /dev/null @@ -1,200 +0,0 @@ -/* remember for development: lines that include the string "@striponbuild" will be stripped on build ;-) */ - -* { cursor: default; } -a { cursor: pointer; } - -html { - direction: ltr; - text-align: left; - overflow: scroll; - overflow-x: hidden; -} - - -html, body { - color: #ccc; - background-color: black; -} - -header { - z-index: 99999; - display: block; - position: fixed; - top: 0ex; - left: 0em; - right: 0em; - height: 5ex; - line-height: 5ex; - text-align: left; - padding: 0ex 1em; - color: #ccc; - background-color: transparent; - cursor: default; - white-space: nowrap; - background-color: rgba(255, 255, 0, 0.5); /* @stripOnBuild */ -} - -header #burger { - color: #eee; - position: fixed; - top: inherit; - right: inherit; - margin-right: 1em; - text-shadow: - -1px -1px 0px black, - 1px -1px 0px black, - -1px 1px 0px black, - 1px 1px 0px black; -} - -header #controls { - text-align: center; - padding-left: 3em; - padding-right: 3em; - position: absolute; - top: -7ex; /* neg. height */ - left: 0em; - right: 0em; - background-color: #111; - background-color: rgba(23, 23, 23, 0.9); - border-bottom: 1px solid #808080; - transition: all 0.6s ease-in-out 0s; -} - -header #controls #serverControls { - margin: 0ex 2em; -} - -header #controls #serverControls button { - margin-right: 0.5em; -} - -header #controls label { - text-align: left; - margin-right: 1em; - margin-left: 1em; -} - -header:hover { - color: white; -} - -header:hover #burger { - font-weight: bold; -} - -header:hover #controls , -#controls.forceShow { - top: 0ex; -} - -#controls.forceShow { - transition: none; -} - -footer { - z-index: 99999; - text-align: center; - position: fixed; - right: 0em; - bottom: 0ex; - left: 0em; - background-color: transparent; - color: #ccc; - font-size: smaller; - height: 4ex; - background-color: rgba(255, 255, 0, 0.5); /* @stripOnBuild */ -} - -#wall article { - position: relative; - margin: 1ex 1ex; - border: 1px solid #999; - display: inline; - display: inline-block; - float: left; -} - -#wall article img { - display: none; -} - -#wall article { - animation: scaleIn linear 0.5s; - -webkit-animation: scaleIn linear 0.5s; - -moz-animation: scaleIn linear 0.5s; - -o-animation: scaleIn linear 0.5s; - -ms-animation: scaleIn linear 0.5s; - - transform-origin: top left; - -webkit-transform-origin: top left; - -moz-transform-origin: top left; - -o-transform-origin: top left; - -ms-transform-origin: top left; -} - - -@keyframes scaleIn { - from { transform: scale(0); } - to { transform: scale(1); } -} - -@-webkit-keyframes scaleIn { - from { -webkit-transform: scale(0); } - to { -webkit-transform: scale(1); } -} - -@-moz-keyframes scaleIn { - from { -moz-transform: scale(0); } - to { -moz-transform: scale(1); } -} - -@-o-keyframes scaleIn { - from { -o-transform: scale(0); } - to { -o-transform: scale(1); } -} - -@-ms-keyframes scaleIn { - from { -ms-transform: scale(0); } - to { -ms-transform: scale(1); } -} - -#wall article img { - display: block; /* needed for alignment of the image to its container box ... */ -} - -#wall article section[role="source"] { - display: block; - position: absolute; - bottom: 0ex; - right: 0em; - line-height: 2.5ex; - min-height: 2.5ex; - min-width: 2.5ex; - background-size: 2.5ex 2.5ex; - background-repeat: no-repeat; - background-position: right bottom; - background-color: transparent; - color: white; - opacity: 0.5; - text-align: initial; /* fall back to native */ -} - -#wall article section[role="source"] a { - display: none; - word-wrap: break-word; - text-decoration: none; - color: inherit; - margin: 0.1ex 0.1em; -} - -#wall article section[role="source"]:hover { - opacity: 1; - background-color: #606060; - background-color: rgba(100, 100, 100, 0.7); - left: 0em; -} - -#wall article section[role="source"]:hover a { - display: block; -} diff --git a/templates_raw/root/css/look_bossMode.css b/templates_raw/root/css/look_bossMode.css deleted file mode 100644 index f20bfd6a..00000000 --- a/templates_raw/root/css/look_bossMode.css +++ /dev/null @@ -1,7 +0,0 @@ -/* remember for development: lines that include the string "@stripOnBuild" will be stripped on build ;-) */ - -/* @TODO write a good looking bos mode that is not totally suspicious */ - -html.boss * { - visibility: hidden; -} diff --git a/templates_raw/root/css/look_buttons.css b/templates_raw/root/css/look_buttons.css deleted file mode 100644 index 784cfc79..00000000 --- a/templates_raw/root/css/look_buttons.css +++ /dev/null @@ -1,43 +0,0 @@ -button , -.button { - min-height: 2ex; - min-width: 2.2ex; - vertical-align: middle; - background-color: #111; - color: #eee; - border: 1px solid #ccc; - border-radius: 0.5ex; - - display: inline; - display: inline-block; - padding: 0.2ex 0.4ex; - - line-height: 2.8ex; -} - - -button:hover , button:active , button:focus , -a.button:hover , a.button:active , a.button:focus , -label.button:hover , label.button:active , label.button:focus { - outline: 0 -} - - -button:hover , button:active -a.button:hover , a.button:active -label.button:hover , label.button:active { - /* invert colors */ - background-color: #ccc; - color: #111; - cursor: pointer; -} - - - -button[disabled] , -.button[disabled] { - opacity: 0.4; - filter: alpha(opacity=40); /* For IE8 and earlier */ - - cursor: default !important; -} diff --git a/templates_raw/root/css/look_hotkeys.css b/templates_raw/root/css/look_hotkeys.css deleted file mode 100644 index 8541f973..00000000 --- a/templates_raw/root/css/look_hotkeys.css +++ /dev/null @@ -1,60 +0,0 @@ - -/* remember for development: lines that include the string "@stripOnBuild" will be stripped on build ;-) */ - -#keys { - position: absolute; - right: 1em; - bottom: 1.2ex; -} - -#hotkeys { - padding: 0.5ex 1em; - line-height: 3ex; - - position: absolute; - left: 0em; - bottom: -5ex; /* neg. height */ - right: 0em; - background-color: #111; - background-color: rgba(23, 23, 23, 0.9); - border-top: 1px solid #808080; - transition: all 0.6s ease-in-out 0s; - - display: inline; - display: inline-block; - margin: 0ex 0em; - list-style: none; -} - -footer:hover #hotkeys , -#hotkeys.forceShow { - bottom: 0ex; -} - -#hotkeys.forceShow { - transition: none; -} - -#hotkeys dd , -#hotkeys dt , -.hotkey { - display: inline; - display: inline-block; - padding: 0.2ex 0.4ex; - margin: 0ex 0em; -} - -#hotkeys dd.active { - background-color: white; - color: black; -} - -#hotkeys dt { - margin-right: 1em; -} - - - - - - diff --git a/templates_raw/root/css/look_stateSwitch.css b/templates_raw/root/css/look_stateSwitch.css deleted file mode 100644 index 2eba86b1..00000000 --- a/templates_raw/root/css/look_stateSwitch.css +++ /dev/null @@ -1,24 +0,0 @@ - -/* remember for development: lines that include the string "@stripOnBuild" will be stripped on build ;-) */ - -.stateSwitchContainer > input { - display: none; -} - -.stateSwitchContainer:after { - display: inline-block; - width: 1.5em; -} - -.stateSwitchContainer.state_0:after { - content: 'off'; -} - -.stateSwitchContainer.state_1:after { - content: 'on'; -} - -.stateSwitchContainer { - cursor: pointer; -} - diff --git a/templates_raw/root/css/normalize.css b/templates_raw/root/css/normalize.css deleted file mode 120000 index 976d81f2..00000000 --- a/templates_raw/root/css/normalize.css +++ /dev/null @@ -1 +0,0 @@ -../../_foreign/normalize.css/normalize.css \ No newline at end of file diff --git a/templates_raw/root/css/sourceIcons.css b/templates_raw/root/css/sourceIcons.css deleted file mode 100644 index 64ece209..00000000 --- a/templates_raw/root/css/sourceIcons.css +++ /dev/null @@ -1,43 +0,0 @@ -/* here are the source favicons defined */ - - -/* resolution is as given, no transformation needed */ -/* used http://www.base64-image.de/ */ - -/* naming conventions: lowercase the names of the crawler classes */ - -#wall article section[role="source"] { /* fallback */ - background-image: url(); -} - -#wall article section[role="source"][data-crawler='pr0gramm'] { - background-image: url(); -} - -#wall article section[role="source"][data-crawler='reddit'] { - background-image: url(); -} - -#wall article section[role="source"][data-crawler='imgur'] { - background-image: url(); -} - -#wall article section[role="source"][data-crawler='soupio'] { - background-image: url(); -} - -#wall article section[role="source"][data-crawler='ninegag'] { - background-image: url(''); -} - -#wall article section[role="source"][data-crawler='instagram'] { - background-image: url(''); -} - -#wall article section[role="source"][data-crawler='fourchan'] { - background-image: url(''); -} - -#wall article section[role="source"][data-crawler='giphy'] { - background-image: url(''); -} diff --git a/templates_raw/root/flush b/templates_raw/root/flush deleted file mode 100644 index 8da85a22..00000000 --- a/templates_raw/root/flush +++ /dev/null @@ -1 +0,0 @@ -3000 \ No newline at end of file diff --git a/templates_raw/root/get b/templates_raw/root/get deleted file mode 100644 index 29d80455..00000000 --- a/templates_raw/root/get +++ /dev/null @@ -1,7 +0,0 @@ -{ - "uri": "./test/testimg.png", - "crawler": "noCrawler", - "site": "get_local", - "source": "http://nicht.parasoup.de/demo/", - "more": {} -} \ No newline at end of file diff --git a/templates_raw/root/js/feel.js b/templates_raw/root/js/feel.js deleted file mode 100644 index 488131cd..00000000 --- a/templates_raw/root/js/feel.js +++ /dev/null @@ -1,485 +0,0 @@ -; /* remember for development: lines that include the string "@stripOnBuild" will be stripped on build ;-) */ - -(function(np, window, undefined) -{ "use strict"; - - var log = window.helperFuncs.log; - log("feel started"); // @stripOnBuild - - var addEvent = window.helperFuncs.addEvent - , fireEvent = window.helperFuncs.fireEvent - ; - - /* - * put everything public/protected in a single-use object called "np" - * every thing else is private ... - */ - - { // some helper variables and shortcuts - var document = window.document - , localStorage = window.helperFuncs.storageFactory() - ; - } - - { // some helper functions - var bitset = { // BitSet helper functions - "gen": function (bit) - { - var r = 1 << bit; - // log("BS gen", bit ,"->", r); // @stripOnBuild - return r; - } - , "check": function (int, bit) - { - var tar = this.gen(bit) - , r = ( (int & tar) == tar ); - // log("BS check", int, bit, " against:", tar, "->", r); // @stripOnBuild - return r; - } - , "set": function (int, bit) - { - var r = int | this.gen(bit); - // log("BS set", int, bit, "->", r); - return r; - } - , "unset": function (int, bit) - { - var r = this.check(int, bit) ? int ^ this.gen(bit) : int; - // log("BS unset", int, bit, "->", r); - return r; - } - }; - } - - np.constants = { // a bunch of public constants - stateBS : { // BitSet positions for states - init : 0 , // init play blocker - manual : 1 , // manual play/pause - boss : 2 , // boss mode - not implemented yet - presented : 3 , // tab is presented - active : 4 , // window is presented - scroll : 5 , // window is scrolled away from top - gallery : 6 // image is clicked for gallery-view - } - }; - - np.__bossMode_className = " boss"; - np.__bossMode_className_RE = new RegExp(np.__bossMode_className, "g"); - - np._imageTarget = undefined; - - np._imageFadeInTime = 1000; - - np._images = []; - np._imagesMax = 50; - - np._state = bitset.set(0, np.constants.stateBS.init); - - np._fetchRequest = new XMLHttpRequest(); - np._serverResetRequest = new XMLHttpRequest(); - np._serverFlushRequest = new XMLHttpRequest(); - - - np._options = { - interval : 10 - , playInBackground : false - }; - - np.__intervall = 0; - - np._setTimer = function(){ - - log("Timer refresh called"); // @stripOnBuild - - if(np.__intervall) - { - window.clearInterval(np.__intervall); - log("Timer Cleared"); // @stripOnBuild - } - - if(this._state == 0) { - np.__intervall = window.setInterval(function () { - np._fetch(); - }, np._options.interval * 1000); - log("Timer Set"); // @stripOnBuild - } - }; - - addEvent(np._fetchRequest, "readystatechange", function () - { - var req = this, imageData = undefined; - log("XHR onreadystatechange", req.readyState); // @stripOnBuild - if ( req.readyState == 4 && req.status == 200) - { - try - { - imageData = JSON.parse(req.responseText); - } - catch (e) - { - log("JSON.parse error", e, req.responseText); // @stripOnBuild - } - if (imageData) - { - np._pushImage(imageData); - } - } - }); - - np.__controllableRequestReadystatechange = function () - { - var req = this; - log("XHR onreadystatechange", req.readyState); // @stripOnBuild - if ( req.readyState == 4 && req.status == 200 ) - { - var controlElement = req.controlElement; - if ( controlElement ) - { - var sleep = parseFloat(req.responseText); - if (isNaN(sleep)) { - sleep = 500; // half a second should be enough as a default ... - } - window.setTimeout(function () - { - controlElement.disabled = false; - controlElement.setAttribute("disabled", null); - controlElement.removeAttribute("disabled"); - }, sleep); - } - } - }; - - addEvent(np._serverResetRequest, "readystatechange", np.__controllableRequestReadystatechange); - addEvent(np._serverFlushRequest, "readystatechange", np.__controllableRequestReadystatechange); - - np._fetch = function () - { - var req = this._fetchRequest; - var r_rs = req.readyState; - if ( r_rs == 4 || r_rs == 0 ) - { - log("triggered fetch"); // @stripOnBuild - req.open("GET", "./get", true); - req.send(); - } - }; - - np._mkImage = function (imageData, onReady) - { - log("mkImage", imageData); // @stripOnBuild - if (! imageData.uri) { return; } - - var imageDoc = document.createElement("img"); - addEvent(imageDoc, "load", function () - { - log("loaded", this.src); // @stripOnBuild - /* structure looks like : - - */ - - var imageBox = document.createElement("article"); - imageBox.appendChild(imageDoc); - - var srcSpan = imageBox.appendChild(document.createElement("section")); - srcSpan.setAttribute("role", "source"); - - // not all browsers support dataset property - so use setAttribute function - srcSpan.setAttribute("data-crawler", imageData.crawler.toLowerCase()); // naming conventions: lowercase the names of the crawler classes - srcSpan.setAttribute("data-site", imageData.site); - if ( imageData.source ) - { // this one is optional - srcSpan.setAttribute("data-source", imageData.source); - } - - var srcA = srcSpan.appendChild(document.createElement("a")); - srcA.href = srcA.innerHTML = srcA.innerText = imageData.source || this.src; - - if ( typeof onReady == "function" ) - { - onReady(imageBox); - } - }); - imageDoc.src = imageData.uri; - }; - - np._pushImage = function (imageData) - { - if ( this._imageTarget ) - { - log("add image", imageData); // @stripOnBuild - this._mkImage(imageData, function (image) - { - var add = ( np._state == 0 ); - if ( add ) - { - np._images.push(image); - np._imageTarget.insertBefore(image, np._imageTarget.firstChild); - log("added image", image); // @stripOnBuild - if ( np._images.length > np._imagesMax ) - { - np._popImage(); - } - } - else // @stripOnBuild - { // @stripOnBuild - log("image not loaded, since _state != 0", np._state); // @stripOnBuild - } // @stripOnBuild - }); - } - else // @stripOnBuild - { // @stripOnBuild - log("! image not added", "no target", this._imageTarget); // @stripOnBuild - } // @stripOnBuild - }; - - np._popImage = function () - { - var image = this._images.shift(); - if ( image && image.parentNode ) - { - image.parentNode.removeChild(image); - } - log("popped an image ... if there was one ... "); // @stripOnBuild - }; - - np._optionsStorage = { - target : "np_store" - , save : function () - { - log("options saved"); - localStorage.setItem(this.target, JSON.stringify(np._options)); - } - , load : function () - { - var lo = localStorage.getItem(this.target); - if ( lo ) - { - try - { - lo = JSON.parse(lo); - log("options loaded"); // @stripOnBuild - np._options = lo; - } - catch ( ex ) - { - log("ERROR:", ex); // @stripOnBuild - } - } - else // @stripOnBuild - { // @stripOnBuild - log("options load failed"); // @stripOnBuild - } // @stripOnBuild - } - }; - - np.serverReset = function (controlElement) - { - var req = this._serverResetRequest; - var r_rs = req.readyState; - if ( r_rs == 4 || r_rs == 0 ) - { - log("triggered serverReset"); // @stripOnBuild - req.controlElement = controlElement; - controlElement.disabled = true; - controlElement.setAttribute("disabled", "disabled"); - req.open("GET", "./reset", true); - req.send(); - } - }; - - np.serverFlush = function (controlElement) - { - var req = this._serverFlushRequest; - var r_rs = req.readyState; - if ( r_rs == 4 || r_rs == 0 ) - { - log("triggered serverFlush"); // @stripOnBuild - req.controlElement = controlElement; - controlElement.disabled = true; - controlElement.setAttribute("disabled", "disabled"); - req.open("GET", "./flush", true); - req.send(); - } - }; - - np.setInterval = function (interval) - { - log("set interval", interval); // @stripOnBuild - this._options.interval = interval; - this._optionsStorage.save(); - np._setTimer(); - }; - - np.getInterval = function () - { - return this._options.interval; - }; - - np.setState = function (which, status) - { - var oldState0 = ( this._state == 0 ); - this._state = bitset[ status ? "set" : "unset" ](this._state, which); - log("setState", which, status, "->", this._state); // @stripOnBuild - var state0 = ( this._state == 0 ); - - // not sure it this is the right place for this ... but i had to put it somewhere ... - if ( which == this.constants.stateBS.boss ) - { - var rootElem = document.documentElement; - if ( status ) - { - rootElem.className += this.__bossMode_className; - try - { - window.blur(); - } - catch ( ex ) - { - // nothing to do ... . - } - } - else - { - rootElem.className = rootElem.className.replace(this.__bossMode_className_RE, ""); - } - } - - if ( state0 != oldState0 ) - { - // state changed - this._setTimer(); - - if (this._state == 0) - { - log("! continue"); // @stripOnBuild - this._fetch(); - } - else - { - log("! halt"); // @stripOnBuild - this._fetchRequest.abort(); - } - } - }; - - np.getState = function (which) - { - return bitset.check(this._state, which); - }; - - np.__inited = false; - np.init = function (imageTargetID, imageFadeInTime) - { - if ( this.__inited ) - { - log("init ran already"); // @stripOnBuild - return false; - } - - this.__inited = true; - log("init started"); // @stripOnBuild - - this._imageTarget = document.getElementById(imageTargetID); - this._imageTarget.appendChild(document.createTextNode("")); // to prevent issues with insertBefore() - log("imageTarget:", this._imageTarget); // @stripOnBuild - - this._imageFadeInTime = imageFadeInTime; - log("imageFadeInTime", this._imageFadeInTime); // @stripOnBuild - - this._optionsStorage.load(); - - - addEvent(window, "scroll", function () - { - log("scroll detected", "offset:", this.pageYOffset); // @stripOnBuild - np.setState(np.constants.stateBS.scroll, this.pageYOffset > 0 ); - }); - this.setState(this.constants.stateBS.scroll, window.pageYOffset > 0 ); - - - var c_background = document.getElementById("c_background"); - c_background.checked = this._options.playInBackground; - fireEvent(c_background, "change"); - addEvent(c_background, "change", function () - { - np._options.playInBackground = this.checked; - np._optionsStorage.save(); - log("playInBackground", np._options.playInBackground); // @stripOnBuild - }); - - - /* blur and focus event is broken some time - can not reproduce :-( */ - addEvent(window, "blur", function () - { - log("!# window blur"); // @stripOnBuild - if ( ! np._options.playInBackground ) - { - np.setState(np.constants.stateBS.active, true); - } - }); - addEvent(window, "focus", function () - { - log("!# window active"); // @stripOnBuild - np.setState(np.constants.stateBS.active, false); - }); - - if ( document.hidden != undefined ) - { - // document. // scroll to top - addEvent(document, "visibilitychange", function () - { - log("-- visibility change detected", "hidden:", this.hidden); // @stripOnBuild - np.setState(np.constants.stateBS.presented, this.hidden); - }); - this.setState(this.constants.stateBS.presented, document.hidden); - } - - var c_speed = document.getElementById("c_speed"); - c_speed.value = this.getInterval(); - addEvent(c_speed, "change", function () - { - np.setInterval(this.value); - log("-- interval changed to", "this.value:", this.value); // @stripOnBuild - }); - - var c_state = document.getElementById("c_state"); - c_state.checked = ! this.getState(this.constants.stateBS.manual); - addEvent(c_state, "change", function () - { - np.setState(np.constants.stateBS.manual, !this.checked); - }); - - - var sc_reset = document.getElementById("sc_reset"); - addEvent(sc_reset, "click", function () - { - var controlElement = this; - if ( ! controlElement.disabled ) - { - np.serverReset(controlElement); - } - }); - - var sc_flush = document.getElementById("sc_flush"); - addEvent(sc_flush, "click", function () - { - var controlElement = this; - if ( ! controlElement.disabled ) - { - np.serverFlush(controlElement); - } - }); - - this.setState(this.constants.stateBS.init, false); - log("init ended"); // @stripOnBuild - }; - - - - log("feel done"); // @stripOnBuild -})(window.nichtparasoup={}, window); diff --git a/templates_raw/root/js/feel_hotkeys.js b/templates_raw/root/js/feel_hotkeys.js deleted file mode 100644 index b3036a34..00000000 --- a/templates_raw/root/js/feel_hotkeys.js +++ /dev/null @@ -1,122 +0,0 @@ -; /* remember for development: lines that include the string "@stripOnBuild" will be stripped on build ;-) */ - - -/* hot keys coming soon. - -idea: -* right(39) := make interval slower -* left(37) := make interval faster -* space(32) := toggle play/pause -* escape(27) := toggle boss mode -> stop all loading, make all content invisible and show some productive instead ;-) - - -if a hot key is pressed, show a badge or something for notification .... -maybe its enough to just toggle the
or something? - -hot keys need to be marked in the