--- /dev/null
+# -*- coding: utf-8 -*-
+#
+# This file is part of Librarian, licensed under GNU Affero GPLv3 or later.
+# Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
+#
+from lxml import etree
+from librarian import OutputFile, RDFNS, DCNS
+from xmlutils import Xmill, tag, tagged, ifoption
+
+
+class EduModule(Xmill):
+ def __init__(self, *args):
+ super(EduModule, self).__init__(*args)
+ self.activity_counter = 0
+ self.question_counter = 0
+
+ def handle_utwor(self, element):
+ v = {}
+# from pdb import *; set_trace()
+ v['title'] = element.xpath('//dc:title/text()', namespaces={'dc':DCNS.uri})[0]
+ return u"""
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="utf-8"/>
+<title>%(title)s</title>
+<link rel="stylesheet" type="text/css" href="master.book.css"/>
+<script src="http://ajax.googleapis.com/ajax/libs/jquery/1.8.3/jquery.min.js"></script>
+<script src="edumed.js"></script>
+</head>
+<body>
+""" % v, u"""
+</body>
+</html>
+
+"""
+
+
+ def handle_powiesc(self, element):
+ return u"""
+<div class="module" id="book-text">
+ <span class="teacher-toggle">
+ <input type="checkbox" name="teacher-toggle" id="teacher-toggle"/>
+ <label for="teacher-toggle">Pokaż treść dla nauczyciela</label>
+ </span>
+
+""", u"</div>"
+
+
+ handle_autor_utworu = tag("span", "author")
+ handle_nazwa_utworu = tag("h1", "title")
+ handle_dzielo_nadrzedne = tag("span", "collection")
+ handle_podtytul = tag("span", "subtitle")
+ handle_naglowek_akt = handle_naglowek_czesc = handle_srodtytul = tag("h2")
+ handle_naglowek_scena = handle_naglowek_rozdzial = tag('h3')
+ handle_naglowek_osoba = handle_naglowek_podrozdzial = tag('h4')
+ handle_akap = handle_akap_dialog = handle_akap_cd = tag('p', 'paragraph')
+ handle_strofa = tag('div', 'stanza')
+
+ def handle_aktywnosc(self, element):
+ self.activity_counter += 1
+ self.options = {
+ 'activity': True,
+ 'activity_counter': self.activity_counter
+ }
+ submill = EduModule()
+
+ opis = submill.generate(element.xpath('opis')[0])
+
+ n = element.xpath('wskazowki')
+ if n: wskazowki = submill.generate(n[0])
+
+ else: wskazowki = ''
+ n = element.xpath('pomoce')
+
+ if n: pomoce = submill.generate(n[0])
+ else: pomoce = ''
+
+ forma = ''.join(element.xpath('forma/text()'))
+
+ czas = ''.join(element.xpath('czas/text()'))
+
+ counter = self.activity_counter
+
+ return u"""
+<div class="activity">
+ <div class="text">%(counter)d.
+ %(opis)s
+ %(wskazowki)s
+ </div>
+ <div class="info">
+ <p>Czas: %(czas)s min</p>
+ <p>Forma: %(forma)s</p>
+ %(pomoce)s
+ </div>
+ <div class="clearboth"></div>
+</div>
+""" % locals()
+
+ handle_opis = ifoption(activity=False)(tag('div', 'description'))
+ handle_wskazowki = ifoption(activity=False)(tag('div', ('hints', 'teacher')))
+
+ @ifoption(activity=False)
+ @tagged('div', 'materials')
+ def handle_pomoce(self, _):
+ return "Pomoce: ", ""
+
+ def handle_czas(self, *_):
+ return
+
+ def handle_forma(self, *_):
+ return
+
+ def handle_cwiczenie(self, element):
+ self.options = {'excercise': element.attrib['typ']}
+ self.question_counter = 0
+ self.piece_counter = 0
+
+ return u"""
+<div class="excercise %(typ)s" data-type="%(typ)s">
+<form action="#" method="POST">
+""" % element.attrib, \
+u"""
+<div class="buttons">
+<span class="message"></span>
+<input type="button" class="check" value="sprawdź"/>
+<input type="button" class="solutions" value="pokaż rozwiązanie"/>
+</div>
+</form>
+</div>
+"""
+
+ def handle_pytanie(self, element):
+ self.question_counter += 1
+ self.piece_counter = 0
+ solution = element.attrib.get('rozw', None)
+ if solution: solution_s = ' data-solution="%s"' % solution
+ else: solution_s = ''
+
+ return '<div class="question" data-no="%d" %s>' %\
+ (self.question_counter, solution_s), \
+ "</div>"
+
+ # Lists
+ def handle_lista(self, element):
+ ltype = element.attrib.get('typ', 'punkt')
+ if ltype == 'slowniczek':
+ self.options = {'slowniczek': True}
+ return '<div class="slowniczek">', '</div>'
+### robie teraz punkty wyboru
+ listtag = {'num': 'ol',
+ 'punkt': 'ul',
+ 'alfa': 'ul',
+ 'czytelnia': 'ul'}[ltype]
+
+ return '<%s class="lista %s">' % (listtag, ltype), '</%s>' % listtag
+
+ def handle_punkt(self, element):
+ if self.options['excercise'] and element.attrib['nazwa']:
+ qc = self.question_counter
+ self.piece_counter += 1
+ no = self.piece_counter
+
+ return u"""
+<li class="question-piece" data-qc="%(qc)d" data-no="%(no)d"><input type="checkbox" name="q%(qc)d_%(no)d"/>
+""" % locals(), u"</li>"
+
+ elif self.options['slowniczek']:
+ return '<dl>', '</dl>'
+ else:
+ return '<li>', '</li>'
+
+ def handle_rdf__RDF(self, _):
+ # ustal w opcjach rzeczy :D
+ return
+
+
+def transform(wldoc, stylesheet='edumed', options=None, flags=None):
+ """Transforms the WL document to XHTML.
+
+ If output_filename is None, returns an XML,
+ otherwise returns True if file has been written,False if it hasn't.
+ File won't be written if it has no content.
+ """
+
+ edumod = EduModule(options)
+# from pdb import set_trace; set_trace()
+ html = edumod.generate(wldoc.edoc.getroot())
+
+ return OutputFile.from_string(html.encode('utf-8'))
--- /dev/null
+# -*- coding: utf-8 -*-
+#
+# This file is part of Librarian, licensed under GNU Affero GPLv3 or later.
+# Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
+#
+from lxml import etree
+from collections import defaultdict
+
+
+class Xmill(object):
+ """Transforms XML to some text.
+ Used instead of XSLT which is difficult and cumbersome.
+
+ """
+ def __init__(self, options=None):
+ self._options = []
+ if options:
+ self._options.append(options)
+
+ def generate(self, document):
+ """Generate text from node using handlers defined in class."""
+ output = self._handle_element(document)
+ return u''.join([x for x in flatten(output) if x is not None])
+
+ @property
+ def options(self):
+ """Returnes merged scoped options for current node.
+ """
+ # Here we can see how a decision not to return the modified map
+ # leads to a need for a hack.
+ return reduce(lambda a, b: a.update(b) or a, self._options, defaultdict(lambda: False))
+
+ @options.setter
+ def options(self, opts):
+ """Sets options overrides for current and child nodes
+ """
+ self._options.append(opts)
+
+
+ def _handle_for_element(self, element):
+ ns = None
+ tagname = None
+# from nose.tools import set_trace
+
+ if isinstance(element, etree._Comment): return None
+
+ if element.tag[0] == '{':
+ for nshort, nhref in element.nsmap.items():
+ try:
+ if element.tag.index('{%s}' % nhref) == 0:
+ ns = nshort
+ tagname = element.tag[len('{%s}' % nhref):]
+ break
+ except ValueError:
+ pass
+ if not ns:
+ raise ValueError("Strange ns for tag: %s, nsmap: %s" %
+ (element.tag, element.nsmap))
+ else:
+ tagname = element.tag
+
+ if ns:
+ meth_name = "handle_%s__%s" % (ns, tagname)
+ else:
+ meth_name = "handle_%s" % (tagname,)
+
+ handler = getattr(self, meth_name, None)
+ return handler
+
+ def next(self, element):
+ if len(element):
+ return element[0]
+
+ while True:
+ sibling = element.getnext()
+ if sibling is not None: return sibling # found a new branch to dig into
+ element = element.getparent()
+ if element is None: return None # end of tree
+
+ def _handle_element(self, element):
+ handler = self._handle_for_element(element)
+ # How many scopes
+ try:
+ options_scopes = len(self._options)
+
+ if handler is None:
+ pre = [element.text]
+ post = []
+ else:
+ vals = handler(element)
+ # depending on number of returned values, vals can be None, a value, or a tuple.
+ # how poorly designed is that? 9 lines below are needed just to unpack this.
+ if vals is None:
+ return []
+ else:
+ if not isinstance(vals, tuple):
+ pre = [vals]
+ post = []
+ else:
+ pre = [vals[0], element.text]
+ post = [vals[1]]
+
+ if element.tail:
+ post.append(element.tail)
+
+ out = pre + [self._handle_element(child) for child in element] + post
+ finally:
+ # clean up option scopes if necessary
+ self._options = self._options[0:options_scopes]
+ return out
+
+
+def tag(name, classes=None, **attrs):
+ """Returns a handler which wraps node contents in tag `name', with class attribute
+ set to `classes' and other attributes according to keyword paramters
+ """
+ if classes:
+ if isinstance(classes, (tuple, list)): classes = ' '.join(classes)
+ attrs['class'] = classes
+ a = ''.join([' %s="%s"' % (k,v) for (k,v) in attrs.items()])
+ def _hnd(self, element):
+ return "<%s%s>" % (name, a), "</%s>" % name
+ return _hnd
+
+
+def tagged(name, classes=None, **attrs):
+ """Handler decorator which wraps handler output in tag `name', with class attribute
+ set to `classes' and other attributes according to keyword paramters
+ """
+ if classes:
+ if isinstance(classes, (tuple,list)): classes = ' '.join(classes)
+ attrs['class'] = classes
+ a = ''.join([' %s="%s"' % (k,v) for (k,v) in attrs.items()])
+ def _decor(f):
+ def _wrap(self, element):
+ r = f(self, element)
+ if r is None: return
+
+ prepend = "<%s%s>" % (name, a)
+ append = "</%s>" % name
+
+ if isinstance(r, tuple):
+ return prepend + r[0], r[1] + append
+ return prepend + r + append
+ return _wrap
+ return _decor
+
+
+def ifoption(**options):
+ """Decorator which enables node only when options are set
+ """
+ def _decor(f):
+ def _handler(self, *args, **kw):
+ opts = self.options
+ for k, v in options.items():
+ if opts[k] != v:
+ return
+ return f(self, *args, **kw)
+ return _handler
+ return _decor
+
+def flatten(l, ltypes=(list, tuple)):
+ """flatten function from BasicPropery/BasicTypes package
+ """
+ ltype = type(l)
+ l = list(l)
+ i = 0
+ while i < len(l):
+ while isinstance(l[i], ltypes):
+ if not l[i]:
+ l.pop(i)
+ i -= 1
+ break
+ else:
+ l[i:i + 1] = l[i]
+ i += 1
+ return ltype(l)