From e4ffa9f5172b88a5a49ed832634105dd65b9727f Mon Sep 17 00:00:00 2001 From: Jan Szejko <janek37@gmail.com> Date: Thu, 8 Mar 2018 15:28:47 +0100 Subject: [PATCH] weasyprint support - in progress --- librarian/parser.py | 4 + librarian/weasy.py | 729 +++++++++++++++++++++++++++++ librarian/weasy/activity-kind.svg | 85 ++++ librarian/weasy/activity-time.svg | 89 ++++ librarian/weasy/activity-tools.svg | 79 ++++ librarian/weasy/bullet.svg | 67 +++ librarian/weasy/weasy.css | 506 ++++++++++++++++++++ 7 files changed, 1559 insertions(+) create mode 100644 librarian/weasy.py create mode 100644 librarian/weasy/activity-kind.svg create mode 100644 librarian/weasy/activity-time.svg create mode 100644 librarian/weasy/activity-tools.svg create mode 100644 librarian/weasy/bullet.svg create mode 100644 librarian/weasy/weasy.css diff --git a/librarian/parser.py b/librarian/parser.py index 12fffb3..d515954 100644 --- a/librarian/parser.py +++ b/librarian/parser.py @@ -171,6 +171,10 @@ class WLDocument(object): from librarian import pyhtml as html return html.transform(self, *args, **kwargs) + def as_weasy(self, *args, **kwargs): + from librarian import weasy + return weasy.transform(self, *args, **kwargs) + def as_text(self, *args, **kwargs): from librarian import text return text.transform(self, *args, **kwargs) diff --git a/librarian/weasy.py b/librarian/weasy.py new file mode 100644 index 0000000..d2cd7e4 --- /dev/null +++ b/librarian/weasy.py @@ -0,0 +1,729 @@ +# -*- 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. +# +import os +import shutil +from tempfile import mkdtemp, NamedTemporaryFile + +from lxml import etree +from subprocess import call + +from librarian import IOFile, Format, ParseError, get_resource +from xmlutils import Xmill, tag, tagged, ifoption, tag_open_close +from librarian import functions +import re +import random +from copy import deepcopy + +IMAGE_THUMB_WIDTH = 300 + + +class EduModule(Xmill): + def __init__(self, options=None): + super(EduModule, self).__init__(options) + self.activity_counter = 0 + self.activity_last = None + self.exercise_counter = 0 + + # text filters + def swap_endlines(txt): + if self.options['strofa']: + txt = txt.replace("/\n", "<br/>\n") + return txt + self.register_text_filter(functions.substitute_entities) + self.register_escaped_text_filter(swap_endlines) + + @tagged('div', 'stanza') + def handle_strofa(self, element): + self.options = {'strofa': True} + return "", "" + + def handle_powiesc(self, element): + return u"""<!DOCTYPE html> +<html> + <head> + <meta http-equiv="content-type" content="text/html; charset=UTF-8"> + <title>Edukacja medialna</title> + <link href="weasy.css" rel="stylesheet" type="text/css"> + <meta charset="UTF-8"> + </head> + <body id="body"> + <div class="module" id="book-text"> +""", u""" + </div> + </body> +</html>""" + + handle_autor_utworu = tag("span", "author") + handle_dzielo_nadrzedne = tag("span", "collection") + handle_podtytul = tag("span", "subtitle") + handle_naglowek_akt = handle_naglowek_czesc = handle_srodtytul = tag("h2") + handle_naglowek_scena = tag('h2') + handle_naglowek_osoba = tag('h3') + handle_akap = handle_akap_dialog = handle_akap_cd = tag('p', 'paragraph') + + handle_wyroznienie = tag('em') + handle_tytul_dziela = tag('em', 'title') + handle_slowo_obce = tag('em', 'foreign') + + def naglowek_to_anchor(self, naglowek): + return self.options['urlmapper'].naglowek_to_anchor(naglowek) + + def handle_nazwa_utworu(self, element): + return "<h1 class='title' id='top'>", "</h1>" + + def handle_naglowek_rozdzial(self, element): + return tag_open_close("h2", id=self.naglowek_to_anchor(element)) + + def handle_naglowek_podrozdzial(self, element): + self.activity_counter = 0 + if element.text.strip() == u'Przebieg zajÄÄ': + return tag('h3', 'activities-header')(self, element) + return tag('h3')(self, element) + + def handle_uwaga(self, _e): + return None + + def handle_aktywnosc(self, element): + self.activity_counter += 1 + parity = 'odd' if self.activity_counter % 2 == 1 else 'even' + if self.activity_counter == 1: + parity += ' first' + self.options = { + 'activity': True, + 'activity_counter': self.activity_counter, + } + submill = EduModule(dict(self.options.items() + {'sub_gen': True}.items())) + + if element.xpath('opis'): + opis = submill.generate(element.xpath('opis')[0]) + else: + opis = '' + + 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()')) + get_forma_url = self.options['urlmapper'].get_forma_url + forms = [] + for form_name in forma.split(','): + name = form_name.strip() + url = get_forma_url(name) + if url: + forms.append("<a href='%s'>%s</a>" % (url, name)) + else: + forms.append(name) + forma = ', '.join(forms) + if forma: + forma = '<tr class="infobox kind"><th>Metoda</th><td><p>%s</p></td></tr>' % forma + + czas = ''.join(element.xpath('czas/text()')) + if czas: + czas = '<tr class="infobox time"><th>Czas</th><td><p>%s min</p></td></tr>' % czas + + counter = self.activity_counter + + if element.getnext().tag == 'aktywnosc' or (len(self.activity_last) and self.activity_last.getnext() == element): + counter_html = """<span class="act_counter">%(counter)d.</span>""" % {'counter': counter} + else: + counter_html = '' + + self.activity_last = element + + return ( + u""" +<div class="activity %(parity)s"> + <div class="text"> + <div class="description"> + %(counter_html)s + %(opis)s""" % {'counter_html': counter_html, 'opis': opis, 'parity': parity}, + u"""%(wskazowki)s + </div> + </div> + <table class="info"> + %(czas)s + %(forma)s + %(pomoce)s + </table> + <div class="clearboth"></div> +</div> +""" % {'wskazowki': wskazowki, 'czas': czas, 'forma': forma, 'pomoce': pomoce}) + + handle_opis = ifoption(sub_gen=True)(tag('div', 'desc')) + handle_wskazowki = ifoption(sub_gen=True)(tag('div', ('hints', 'teacher'))) + + @ifoption(sub_gen=True) + @tagged('tr', 'infobox materials') + def handle_pomoce(self, _): + return """<th>Pomoce</th><td>""", "</td>" + + def handle_czas(self, *_): + return + + def handle_forma(self, *_): + return + + def handle_cwiczenie(self, element): + exercise_handlers = { + 'wybor': Wybor, + 'uporzadkuj': Uporzadkuj, + 'luki': Luki, + 'zastap': Zastap, + 'przyporzadkuj': Przyporzadkuj, + 'prawdafalsz': PrawdaFalsz + } + + typ = element.attrib['typ'] + self.exercise_counter += 1 + self.options = {'exercise_counter': self.exercise_counter} + handler = exercise_handlers[typ](self.options) + return handler.generate(element) + + # Lists + def handle_lista(self, element, attrs=None): + if attrs is None: + attrs = {} + ltype = element.attrib.get('typ', 'punkt') + if not element.findall("punkt"): + if ltype == 'czytelnia': + return '<p>W przygotowaniu.</p>' + else: + return None + if ltype == 'slowniczek': + surl = element.attrib.get('src', None) + if surl is None: + # print '** missing src on <slowniczek>, setting default' + surl = 'http://edukacjamedialna.edu.pl/lekcje/slowniczek/' + sxml = etree.fromstring(self.options['provider'].by_uri(surl).get_string()) + + self.options = {'slowniczek': True, 'slowniczek_xml': sxml} + return '<div class="slowniczek"><dl>', '</dl></div>' + + listtag = { + 'num': 'ol', + 'punkt': 'ul', + 'alfa': 'ul', + 'czytelnia': 'ul'}[ltype] + + classes = attrs.get('class', '') + if classes: + del attrs['class'] + + attrs_s = ' '.join(['%s="%s"' % kv for kv in attrs.items()]) + if attrs_s: + attrs_s = ' ' + attrs_s + + return '<%s class="lista %s %s"%s>' % (listtag, ltype, classes, attrs_s), '</%s>' % listtag + + def handle_punkt(self, element): + if self.options['slowniczek']: + return '', '' + else: + return '<li>', '</li>' + + def handle_definiendum(self, element): + nxt = element.getnext() + definiens_s = '' + + if not element.text: + print "!! Empty <definiendum/>" + return None + + # let's pull definiens from another document + if self.options['slowniczek_xml'] is not None and (nxt is None or nxt.tag != 'definiens'): + sxml = self.options['slowniczek_xml'] + if "'" in (element.text or ''): + defloc = sxml.xpath("//definiendum[text()=\"%s\"]" % (element.text or '').strip()) + else: + defloc = sxml.xpath("//definiendum[text()='%s']" % (element.text or '').strip()) + if defloc: + definiens = defloc[0].getnext() + if definiens.tag == 'definiens': + subgen = EduModule(self.options) + definiens_s = subgen.generate(definiens) + else: + print ("!! Missing definiendum in source: '%s'" % element.text).encode('utf-8') + + return u"<dt id='%s'>" % self.naglowek_to_anchor(element), u"</dt>" + definiens_s + + def handle_definiens(self, element): + return u"<dd>", u"</dd>" + + def handle_podpis(self, element): + return u"""<div class="caption">""", u"</div>" + + def handle_tabela(self, element): + has_frames = int(element.attrib.get("ramki", "0")) + frames_c = "framed" if has_frames else "" + return u"""<table class="%s">""" % frames_c, u"</table>" + + def handle_wiersz(self, element): + return u"<tr>", u"</tr>" + + def handle_kol(self, element): + return u"<td>", u"</td>" + + def handle_rdf__RDF(self, _): + # ustal w opcjach rzeczy :D + return + + def handle_link(self, element): + if 'url' in element.attrib: + return tag('a', href=element.attrib['url'])(self, element) + elif 'material' in element.attrib: + material_err = u' [BRAKUJÄCY MATERIAÅ]' + slug = element.attrib['material'] + + def make_url(f): + return self.options['urlmapper'].url_for_material(slug, f) + + formats = self.options['urlmapper'].materials(slug) + + try: + def_href = make_url(formats[0][0]) + def_err = u"" + except (IndexError, self.options['urlmapper'].MaterialNotFound): + def_err = material_err + def_href = u"" + fmt_links = [] + for f in formats[1:]: + try: + fmt_links.append(u'<a href="%s">%s</a>' % (make_url(f[0]), f[0].upper())) + except self.options['urlmapper'].MaterialNotFound: + fmt_links.append(u'<a>%s%s</a>' % (f[0].upper(), material_err)) + more_links = u' (%s)' % u', '.join(fmt_links) if fmt_links else u'' + + return u"<a href='%s'>" % def_href, u'%s</a>%s' % (def_err, more_links) + + def handle_obraz(self, element): + format = self.options['urlmapper'] + name = element.attrib.get('nazwa', '').strip() + if not name: + print '!! <obraz> missing "nazwa"' + return + alt = element.attrib.get('alt', '') + if not alt: + print '** <obraz> missing "alt"' + slug, ext = name.rsplit('.', 1) + image = format.image(slug, ext) + name = image.name.rsplit('/', 1)[-1] + e = etree.Element("a", attrib={"class": "image"}) + e.append(etree.Element("img", attrib={ + "src": name, + "alt": alt, + "width": str(IMAGE_THUMB_WIDTH)})) + format.attachments[name] = self.options['media_root'] + image.name + return etree.tostring(e, encoding=unicode), u"" + + def handle_video(self, element): + url = element.attrib.get('url') + if not url: + print '!! <video> missing url' + return + m = re.match(r'(?:https?://)?(?:www.)?youtube.com/watch\?(?:.*&)?v=([^&]+)(?:$|&)', url) + if not m: + print '!! unknown <video> url scheme:', url + return + return """<iframe width="630" height="384" src="http://www.youtube.com/embed/%s" + frameborder="0" allowfullscreen></iframe>""" % m.group(1), "" + + +class Exercise(EduModule): + INSTRUCTION = "" + + def __init__(self, *args, **kw): + self.question_counter = 0 + super(Exercise, self).__init__(*args, **kw) + self.instruction_printed = False + self.piece_counter = None + + @tagged('div', 'description') + def handle_opis(self, element): + return "", self.get_instruction() + + def handle_rozw_kom(self, element): + return u"""<div style="display:none" class="comment">""", u"""</div>""" + + def extra_attributes(self): + return {} + + def handle_cwiczenie(self, element): + self.options = {'exercise': element.attrib['typ']} + self.question_counter = 0 + self.piece_counter = 0 + + extra_attrs = self.extra_attributes() + + pre = u""" +<div class="exercise %(typ)s" data-type="%(typ)s"%(extra_attrs)s> +<h3>Zadanie %(exercies_counter)d</h3> +""" % { + 'exercies_counter': self.options['exercise_counter'], + 'typ': element.attrib['typ'], + 'extra_attrs': ' ' + ' '.join( + 'data-%s="%s"' % item for item in extra_attrs.iteritems()) if extra_attrs else '', + } + post = u""" +</div> +""" + # Add a single <pytanie> tag if it's not there + if not element.xpath(".//pytanie"): + qpre, qpost = self.handle_pytanie(element) + pre += qpre + post = qpost + post + return pre, post + + def handle_pytanie(self, element): + """This will handle <cwiczenie> element, when there is no <pytanie> + """ + add_class = "" + self.question_counter += 1 + self.piece_counter = 0 + solution = element.attrib.get('rozw', None) + solution_s = ' data-solution="%s"' % solution if solution else '' + + handles = element.attrib.get('uchwyty', None) + if handles: + add_class += ' handles handles-%s' % handles + self.options = {'handles': handles} + + minimum = element.attrib.get('min', None) + minimum_s = ' data-minimum="%d"' % int(minimum) if minimum else '' + + return '<div class="question%s" data-no="%d" %s>' %\ + (add_class, self.question_counter, solution_s + minimum_s), \ + "</div>" + + def get_instruction(self): + if not self.instruction_printed: + self.instruction_printed = True + if self.INSTRUCTION: + return u'<span class="instruction">%s</span>' % self.INSTRUCTION + else: + return "" + else: + return "" + + +class Wybor(Exercise): + def extra_attributes(self): + return {'subtype': 'single' if self.options['single'] else 'multiple'} + + def handle_cwiczenie(self, element): + is_single_choice = True + pytania = element.xpath(".//pytanie") + if not pytania: + pytania = [element] + for p in pytania: + solutions = p.xpath(".//punkt[@rozw='prawda']") + if len(solutions) != 1: + is_single_choice = False + break + + self.options = {'single': is_single_choice} + return super(Wybor, self).handle_cwiczenie(element) + + def handle_punkt(self, element): + if self.options['exercise'] and element.attrib.get('rozw', None): + qc = self.question_counter + self.piece_counter += 1 + no = self.piece_counter + eid = "q%(qc)d_%(no)d" % locals() + sol = element.attrib.get('rozw', None) + params = {'qc': qc, 'no': no, 'sol': sol, 'eid': eid} + if self.options['single']: + input_tag = u'<input type="radio" name="q%(qc)d" id="%(eid)s" value="%(eid)s" />' + else: + input_tag = u'<input type="checkbox" name="%(eid)s" id="%(eid)s" />' + return (u""" +<li class="question-piece" data-qc="%(qc)d" data-no="%(no)d" data-sol="%(sol)s"> + """ + input_tag + u""" +<label for="%(eid)s">""") % params, u"</label></li>" + else: + return super(Wybor, self).handle_punkt(element) + + +class Uporzadkuj(Exercise): + INSTRUCTION = u"Kliknij wybranÄ odpowiedź i przeciÄ gnij w nowe miejsce." + + def handle_pytanie(self, element): + """ +Overrides the returned content default handle_pytanie + """ + # we ignore the result, returning our own + super(Uporzadkuj, self).handle_pytanie(element) + order_items = element.xpath(".//punkt/@rozw") + + return u"""<div class="question" data-original="%s" data-no="%s">""" % \ + (','.join(order_items), self.question_counter), \ + u"""</div>""" + + def handle_punkt(self, element): + return """<li class="question-piece" data-pos="%(rozw)s">""" \ + % element.attrib,\ + "</li>" + + +class Luki(Exercise): + INSTRUCTION = u"PrzeciÄ gnij odpowiedzi i upuÅÄ w wybranym polu." + + def find_pieces(self, question): + return question.xpath(".//luka") + + def solution_html(self, piece): + piece = deepcopy(piece) + piece.tail = None + sub = EduModule() + return sub.generate(piece) + + def handle_pytanie(self, element): + qpre, qpost = super(Luki, self).handle_pytanie(element) + + luki = list(enumerate(self.find_pieces(element))) + luki_html = "" + random.shuffle(luki) + for (i, luka) in luki: + i += 1 + luka_html = self.solution_html(luka) + luki_html += u'<span class="draggable question-piece" data-no="%d">%s</span>' % (i, luka_html) + self.words_html = '<div class="words">%s</div>' % luki_html + + return qpre, qpost + + def handle_opis(self, element): + return '', self.words_html + + def handle_luka(self, element): + self.piece_counter += 1 + return '<span class="placeholder" data-solution="%d"></span>' % self.piece_counter + + +class Zastap(Luki): + INSTRUCTION = u"PrzeciÄ gnij odpowiedzi i upuÅÄ je na sÅowie lub wyrażeniu, które chcesz zastÄ piÄ." + + def find_pieces(self, question): + return question.xpath(".//zastap") + + def solution_html(self, piece): + return piece.attrib.get('rozw', '') + + def handle_zastap(self, element): + self.piece_counter += 1 + return '<span class="placeholder zastap question-piece" data-solution="%d">' \ + % self.piece_counter, '</span>' + + +class Przyporzadkuj(Exercise): + INSTRUCTION = [u"PrzeciÄ gnij odpowiedzi i upuÅÄ w wybranym polu.", + u"Kliknij numer odpowiedzi, przeciÄ gnij i upuÅÄ w wybranym polu."] + + def get_instruction(self): + if not self.instruction_printed: + self.instruction_printed = True + return u'<span class="instruction">%s</span>' % self.INSTRUCTION[self.options['handles'] and 1 or 0] + else: + return "" + + def handle_cwiczenie(self, element): + pre, post = super(Przyporzadkuj, self).handle_cwiczenie(element) + lista_with_handles = element.xpath(".//*[@uchwyty]") + if lista_with_handles: + self.options = {'handles': True} + return pre, post + + def handle_pytanie(self, element): + pre, post = super(Przyporzadkuj, self).handle_pytanie(element) + minimum = element.attrib.get("min", None) + if minimum: + self.options = {"min": int(minimum)} + return pre, post + + def handle_lista(self, lista): + if 'nazwa' in lista.attrib: + attrs = { + 'data-name': lista.attrib['nazwa'], + 'class': 'predicate' + } + self.options = {'predicate': True} + elif 'cel' in lista.attrib: + attrs = { + 'data-target': lista.attrib['cel'], + 'class': 'subject' + } + if lista.attrib.get('krotkie'): + self.options = {'short': True} + self.options = {'subject': True} + else: + attrs = {} + pre, post = super(Przyporzadkuj, self).handle_lista(lista, attrs) + return pre, post + '<br class="clearboth"/>' + + def handle_punkt(self, element): + if self.options['subject']: + self.piece_counter += 1 + if self.options['handles']: + return ( + '<li><span data-solution="%s" data-no="%s" ' + 'class="question-piece draggable handle add-li">%s</span>' % ( + element.attrib.get('rozw', ''), + self.piece_counter, + self.piece_counter), + '</li>') + else: + extra_class = "" + if self.options['short']: + extra_class += ' short' + return '<li data-solution="%s" data-no="%s" class="question-piece draggable%s">' % ( + element.attrib.get('rozw', ''), + self.piece_counter, extra_class), '</li>' + + elif self.options['predicate']: + if self.options['min']: + placeholders = u'<li class="placeholder"></li>' * self.options['min'] + else: + placeholders = u'<li class="placeholder multiple"></li>' + return ( + '<li data-predicate="%s">' % element.attrib.get('nazwa', ''), + '<ul class="subjects">' + placeholders + '</ul></li>') + + else: + return super(Przyporzadkuj, self).handle_punkt(element) + + +class PrawdaFalsz(Exercise): + def handle_punkt(self, element): + if 'rozw' in element.attrib: + return u'''<li data-solution="%s" class="question-piece"> + <span class="buttons"> + <a data-value="true" class="true">Prawda</a> + <a data-value="false" class="false">FaÅsz</a> + </span> + <span class="question-piece-text">''' % { + 'prawda': 'true', + 'falsz': 'false' + }[element.attrib['rozw']], '</span></li>' + else: + return super(PrawdaFalsz, self).handle_punkt(element) + + +class EduModuleWeasyFormat(Format): + PRIMARY_MATERIAL_FORMATS = ('pdf', 'odt') + + class MaterialNotFound(BaseException): + pass + + def __init__(self, wldoc, media_root='', **kwargs): + super(EduModuleWeasyFormat, self).__init__(wldoc, **kwargs) + self.media_root = media_root + self.materials_by_slug = None + self.attachments = {} + + def get_html(self): + self.attachments = {} + edumod = EduModule({ + 'provider': self.wldoc.provider, + 'urlmapper': self, + 'wldoc': self.wldoc, + 'media_root': self.media_root, + }) + return edumod.generate(self.wldoc.edoc.getroot()) + + def get_weasy_dir(self): + html = self.get_html() + temp = mkdtemp('-weasy') + # Save TeX file + html_path = os.path.join(temp, 'doc.html') + with open(html_path, 'w') as fout: + fout.write(html.encode('utf-8')) + # Copy style + weasy_dir = os.path.join(os.path.dirname(__file__), 'weasy') + for filename in os.listdir(weasy_dir): + shutil.copy(get_resource('weasy/%s' % filename), temp) + for name, path in self.attachments.items(): + shutil.copy(path, os.path.join(temp, name)) + return temp + + def get_pdf(self): + temp = self.get_weasy_dir() + html_path = os.path.join(temp, 'doc.html') + pdf_path = os.path.join(temp, 'doc.pdf') + try: + cwd = os.getcwd() + except OSError: + cwd = None + os.chdir(temp) + + WEASY_COMMAND = '/home/janek/Desktop/weasy-test/bin/weasyprint' + + p = call([WEASY_COMMAND, html_path, pdf_path]) + if p: + raise ParseError("Error parsing .html file: %s" % html_path) + + if cwd is not None: + os.chdir(cwd) + + output_file = NamedTemporaryFile(prefix='librarian', suffix='.pdf', delete=False) + shutil.move(pdf_path, output_file.name) + # shutil.rmtree(temp) + return IOFile.from_filename(output_file.name) + + def build(self): + # Sort materials by slug. + self.materials_by_slug = {} + for name, att in self.wldoc.source.attachments.items(): + parts = name.rsplit('.', 1) + if len(parts) == 1: + continue + slug, ext = parts + if slug not in self.materials_by_slug: + self.materials_by_slug[slug] = {} + self.materials_by_slug[slug][ext] = att + return self.get_pdf() + + def materials(self, slug): + """Returns a list of pairs: (ext, iofile).""" + order = {pmf: i for (i, pmf) in enumerate(self.PRIMARY_MATERIAL_FORMATS)} + mats = self.materials_by_slug.get(slug, {}).items() + if not mats: + print ("!! Material missing: '%s'" % slug).encode('utf-8') + return sorted(mats, key=lambda (x, y): order.get(x, x)) + + def url_for_material(self, slug, fmt): + return "%s.%s" % (slug, fmt) + + def url_for_image(self, slug, fmt, width=None): + return self.url_for_material(slug, fmt) + + def text_to_anchor(self, text): + return re.sub(r" +", " ", text) + + def naglowek_to_anchor(self, naglowek): + return self.text_to_anchor(naglowek.text.strip()) + + def get_forma_url(self, forma): + return None + + def get_help_url(self, naglowek): + return None + + +def transform(wldoc, stylesheet='edumed', options=None, flags=None, verbose=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. + """ + edumodfor = EduModuleWeasyFormat(wldoc) + return edumodfor.build() diff --git a/librarian/weasy/activity-kind.svg b/librarian/weasy/activity-kind.svg new file mode 100644 index 0000000..21f93a0 --- /dev/null +++ b/librarian/weasy/activity-kind.svg @@ -0,0 +1,85 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> + +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + id="svg2" + version="1.1" + inkscape:version="0.91 r13725" + width="31.25" + height="30" + viewBox="0 0 31.25 30" + sodipodi:docname="activity-kind.svg"> + <metadata + id="metadata8"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs6" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1366" + inkscape:window-height="692" + id="namedview4" + showgrid="false" + inkscape:zoom="11.125147" + inkscape:cx="1.7375427" + inkscape:cy="15" + inkscape:window-x="0" + inkscape:window-y="24" + inkscape:window-maximized="1" + inkscape:current-layer="svg2" /> + <rect + style="fill:none;fill-opacity:1;stroke:#000000;stroke-width:2;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="rect3338" + width="28.314232" + height="20.044682" + x="1.6179562" + y="2.5846312" + ry="10.022341" /> + <circle + style="fill:#000000;fill-opacity:1;stroke:none;stroke-width:2;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="path4140" + cx="10.741432" + cy="12.247425" + r="1.4831265" /> + <circle + r="1.4831265" + cy="12.247425" + cx="16.40428" + id="circle4144" + style="fill:#000000;fill-opacity:1;stroke:none;stroke-width:2;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" /> + <circle + style="fill:#000000;fill-opacity:1;stroke:none;stroke-width:2;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="circle4146" + cx="21.887354" + cy="12.247425" + r="1.4831265" /> + <path + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" + d="m 6.5617113,26.404542 c 2.0227654,-0.0019 3.9526297,-1.24417 3.9550037,-3.865118" + id="path4148" + inkscape:connector-curvature="0" + sodipodi:nodetypes="cc" /> +</svg> diff --git a/librarian/weasy/activity-time.svg b/librarian/weasy/activity-time.svg new file mode 100644 index 0000000..f9db82e --- /dev/null +++ b/librarian/weasy/activity-time.svg @@ -0,0 +1,89 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> + +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + id="svg2" + version="1.1" + inkscape:version="0.91 r13725" + width="31.25" + height="30" + viewBox="0 0 31.25 30" + sodipodi:docname="activity-time.svg"> + <metadata + id="metadata8"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs6" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1366" + inkscape:window-height="692" + id="namedview4" + showgrid="false" + inkscape:zoom="5.5625733" + inkscape:cx="15.625" + inkscape:cy="25.786375" + inkscape:window-x="0" + inkscape:window-y="24" + inkscape:window-maximized="1" + inkscape:current-layer="svg2" /> + <circle + style="fill:none;fill-opacity:1;stroke:#000000;stroke-width:2;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="path4138" + r="13.093221" + cy="14.745763" + cx="14.872882" /> + <path + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2.31188059;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" + d="m 14.910113,1.9260655 0,2.077502" + id="path4142" + inkscape:connector-curvature="0" + sodipodi:nodetypes="cc" /> + <path + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2.31188059;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" + d="m 4.1847769,14.80919 -2.077502,0" + id="path4142-3" + inkscape:connector-curvature="0" + sodipodi:nodetypes="cc" /> + <path + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2.31188059;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" + d="m 27.645142,14.719303 -2.077502,0" + id="path4142-3-6" + inkscape:connector-curvature="0" + sodipodi:nodetypes="cc" /> + <path + sodipodi:nodetypes="cc" + inkscape:connector-curvature="0" + id="path4174" + d="m 14.910113,25.431374 0,2.077502" + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2.31188059;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" /> + <path + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" + d="m 14.966095,9.0564559 0,6.0223921 5.775205,5.775205" + id="path4176" + inkscape:connector-curvature="0" + sodipodi:nodetypes="ccc" /> +</svg> diff --git a/librarian/weasy/activity-tools.svg b/librarian/weasy/activity-tools.svg new file mode 100644 index 0000000..63cd72a --- /dev/null +++ b/librarian/weasy/activity-tools.svg @@ -0,0 +1,79 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> + +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + id="svg2" + version="1.1" + inkscape:version="0.91 r13725" + width="31.25" + height="30" + viewBox="0 0 31.25 30" + sodipodi:docname="activity-tools.svg"> + <metadata + id="metadata8"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs6" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1366" + inkscape:window-height="692" + id="namedview4" + showgrid="false" + inkscape:zoom="12.136524" + inkscape:cx="10.625" + inkscape:cy="13.75" + inkscape:window-x="0" + inkscape:window-y="24" + inkscape:window-maximized="1" + inkscape:current-layer="svg2" /> + <g + id="g3336" + transform="translate(4.7789631,0)"> + <circle + r="3.596884" + cy="7.3358049" + cx="4.8940678" + id="path4138" + style="fill:none;fill-opacity:1;stroke:#000000;stroke-width:1.91428244;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" /> + <circle + style="fill:none;fill-opacity:1;stroke:#000000;stroke-width:1.93877327;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="circle4140" + cx="16.54661" + cy="7.4523311" + r="3.6429014" /> + <path + inkscape:connector-curvature="0" + id="path4142" + d="M 6.9915254,11.006356 18.527542,28.543432" + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" /> + <path + style="color:#000000;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate" + d="M 14.274364,11.006356 2.7383474,28.543432" + id="path4153" + inkscape:connector-curvature="0" /> + </g> +</svg> diff --git a/librarian/weasy/bullet.svg b/librarian/weasy/bullet.svg new file mode 100644 index 0000000..93038b3 --- /dev/null +++ b/librarian/weasy/bullet.svg @@ -0,0 +1,67 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> + +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="2.5116279mm" + height="4.5116282mm" + viewBox="0 0 8.8994689 15.986082" + id="svg2" + version="1.1" + inkscape:version="0.91 r13725" + sodipodi:docname="bullet.svg"> + <defs + id="defs4" /> + <sodipodi:namedview + id="base" + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1.0" + inkscape:pageopacity="0.0" + inkscape:pageshadow="2" + inkscape:zoom="18.203333" + inkscape:cx="6.1928641" + inkscape:cy="7.9379608" + inkscape:document-units="px" + inkscape:current-layer="layer1" + showgrid="false" + inkscape:window-width="1366" + inkscape:window-height="692" + inkscape:window-x="0" + inkscape:window-y="24" + inkscape:window-maximized="1" + fit-margin-top="2" + fit-margin-left="0" + fit-margin-right="0" + fit-margin-bottom="0" /> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <g + inkscape:label="Layer 1" + inkscape:groupmode="layer" + id="layer1" + transform="translate(0,-1036.376)"> + <circle + style="fill:#cccccc;fill-opacity:1;stroke:none;stroke-width:0.40000001;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + id="path3336" + cx="4.4497347" + cy="1047.9124" + r="4.4497347" /> + </g> +</svg> diff --git a/librarian/weasy/weasy.css b/librarian/weasy/weasy.css new file mode 100644 index 0000000..e0ce17b --- /dev/null +++ b/librarian/weasy/weasy.css @@ -0,0 +1,506 @@ +@charset "UTF-8"; +/*@import url(https://fonts.googleapis.com/css?family=Dosis:400,700&subset=latin,latin-ext);*/ + +@page { + size: A4; + margin: 1.5cm; + @bottom-center { + counter-increment: page; + content: "Strona " counter(page); + font-family: Dosis; + font-size: 12pt; + } +} + +a { + text-decoration: none; + color: #ed7831; } + a img { + width: 8cm; + border: 0; + padding: 0; + display: block; + margin-left: auto; + margin-right: auto; } + +ul { + list-style-image: url('bullet.svg'); +} + +h1, h2, h3 { + page-break-after: avoid; +} + +body { + font-family: Dosis; + background: white; + color: #363a3e; + margin: 0; } + +.link-list { + list-style: none; + padding: 0; } + .link-list li { + margin-bottom: .5em; } + .link-list a:before { + content: "â"; + margin-right: .5em; } + .link-list a { + color: #363a3e; } + .link-list a:hover { + color: #ed7831; } + +.link-list-colored a { + color: #ed7831; } + +.plain { + margin: 0; + padding: 0; + list-style: none; } + .plain li { + margin: 1em 0; } + +.lesson-footer { + clear: both; + border-top: 1px solid #777; + margin-top: 2em; + padding-top: 1em; } + .lesson-footer .section-info { + text-align: center; } + .lesson-footer .previous-lesson { + float: left; } + .lesson-footer .next-lesson { + float: right; } + +ol.alpha { + counter-reset: list; } + ol.alpha > li { + list-style: none; + position: relative; } + ol.alpha > li:before { + counter-increment: list; + content: counter(list,lower-alpha) ") "; + position: absolute; + left: -1.4em; } + +blockquote { + font-size: 0.875em; } + +/* =================== */ +/* = Custom elements = */ +/* =================== */ +div.stanza { + margin: 1.5em 0 0; } + +p.paragraph { + text-align: justify; + margin: 1.5em 0 0; } + +hr.spacer { + height: 3em; + visibility: hidden; } + +hr.spacer-line { + margin: 1.5em 0; + border: none; + border-bottom: 0.1em solid #000; } + +p.spacer-asterisk { + padding: 0; + margin: 1.5em 0; + text-align: center; } + +table.framed { + border-collapse: collapse; +} + +table.framed td, table.framed.th { + border: 1px #888 solid; +} + +.activities-header, #slowniczek { + color: white; + background: #ed7831; + display: inline-block; + padding: 0.1cm 0.3cm; + margin-bottom: -0.1cm; + border-radius: 0.3cm 0.3cm 0 0; + page-break-after: avoid; +} + +#slowniczek { + margin-left: 2cm; +} + +.activity { + clear: both; } + +.activity.first { + margin-top: 0; + page-break-before: avoid; +} + +.activity .text { + padding-top: 0.5cm; +} + +.activity.odd .text { + border: thick solid #ed7831; + margin-right: -3cm; + padding-right: 3cm; + padding-bottom: 1cm; +} + +.slowniczek { + border: thick solid #ed7831; + margin-left: 2cm; + margin-right: -3cm; + padding: 0.5cm 3cm 0.5cm 0.5cm; + page-break-before: avoid; +} + +.activity.even .text { + padding-bottom: 0.5cm; + margin-left: 2cm; +} + +.activity .description { + width: 13cm; + margin-left: 2cm; + position: relative; +} +.activity .description p:first-child { + margin-top: 0; } + +.activity { + margin-top: 2em; } + .activity.odd { + page-break-inside: avoid; } + .activity .act_counter { + position: absolute; + top: 0; + left: -1.2cm; + font-weight: bold; + font-size: 2em; } + .activity.odd .info { + float: right; + position: relative; + top: -0.8cm; + margin-bottom: -0.8cm; + } + .activity .info { + width: 50%; + padding: 0.3cm; + border: thick solid #ed7831; + border-radius: 0.8cm; + page-break-before: avoid; + page-break-inside: avoid; + background-color: white; } + .activity .info .infobox { + padding: 1em 0; + border-top: 1px solid #c9ccce; } + .activity .info .infobox th { + text-transform: uppercase; + margin: 0 0 0.5em -2.1875em; + padding-left: 2.1875em; + line-height: 24px; + font-size: 1em; } + .activity .info .infobox p { + margin: 0; } + .activity table.info { border-spacing: 0.2cm } + .activity .info th, .activity .info td { vertical-align: top; } + .activity .info th { + background: no-repeat 0 0; + background-size: 25px 25px; } + .activity .info .time th { + background-image: url("activity-time.svg"); } + .activity .info .kind th { + background-image: url("activity-kind.svg"); } + .activity .info .materials th { + background-image: url("activity-tools.svg"); } + +.materials-list { + border: thick solid #ed7831; + border-radius: 0.5cm; + width: 60%; + padding: 0 0.3cm 0.3cm; + margin-top: 0.5cm; + page-break-inside: avoid; +} +.materials-list p { + margin-top: 0.2cm; + margin-bottom: 0.2cm; +} + +dl { + width: 100%; + padding: 0; + margin: 0; +} +dt { + display: inline-block; + width: 20%; + padding: 0.1cm; + margin: 0; + font-weight: bold; + vertical-align: top; +} +dd { + display: inline-block; + width: 70%; + padding: 0.1cm; + margin: 0; + vertical-align: top; +} + +.lista .paragraph { + margin: .3em 0; } +.lista li { + margin: .75em 0; } + +.clearboth { + clear: both; } + +#book-text .top-link { + margin-top: 1em; } + +.help { + font-size: .7em; + padding: 0 .5em; + color: #888; + vertical-align: super; } + +.exercise img { + vertical-align: top; } +.exercise .question-piece { + border-radius: 0.6875em; + padding: 0.5em 0.6875em; } +.exercise .lista.punkt { + list-style: none; + margin: 0; + padding: 0; } +.exercise .lista li { + margin: 0.3125em 0; } +.exercise .question-piece.correct { + background-color: #16a487; + color: white; } +.exercise .question-piece.incorrect { + background-color: #f00; + color: white; } +.exercise .placeholder.dragover { + background: #ed7831 !important; } +.exercise .remove { + float: right; + display: inline-block; + border-radius: 0.5em; + padding: 0.25em 0.5em; + margin: -0.25em -0.5em -0.25em 0.5em; + background: #ed7831; + color: white; } +.exercise .buttons { + background: #5e6165; + border-radius: 0.5625em; + height: 1.875em; + line-height: 1.875em; + overflow: hidden; + color: white; + margin-top: 1.875em; } + .exercise .buttons input { + height: 100%; + border: 0; + background: #8e9093; + color: white; + text-transform: uppercase; + font-weight: bold; } + .exercise .buttons .message { + float: right; + padding: 0 1.25em; + background: red; } + .exercise .buttons .maxscore { + background: #16a487; } + .exercise .buttons input.check { + background: #ed7831; } + +.exercise .instruction { + display: block; + margin: .3em; + color: green; + font-size: .9em; } +.exercise .instruction:before { + content: "â "; } + +.luki .question-piece { + background-color: #d4d6d8; + color: #363a3e; + z-index: 2; + display: inline-block; + margin: 0.3125em; + max-width: 38em; } +.luki .question-piece.disabled { + background-color: #eee; + color: #d4d6d8; } +.luki .placeholder { + border-radius: 0.6875em; + padding: 0.5em 0.6875em; + display: inline-block; + margin: 0.3125em; + width: 4em; + background-color: #eee; + z-index: 1; } +.luki .placeholder:after { + content: "\0000a0"; } + +.zastap .question-piece { + background-color: #d4d6d8; + color: #363a3e; + display: inline-block; + margin: 0.3125em; + z-index: 2; } +.zastap .question-piece.disabled { + background-color: #eee; + color: #d4d6d8; } +.zastap .question-piece.placeholder { + background-color: inherit; + color: inherit; + z-index: inherit; + display: inline; + margin: 0; + padding: 0; + border-radius: 0; } + +.uporzadkuj .question-piece { + background-color: #d4d6d8; + color: #363a3e; + z-index: 2; } +.uporzadkuj .question-piece.disabled { + background-color: #eee; + color: #d4d6d8; } +.uporzadkuj .lista { + list-style: none; + margin: 0; + padding: 0; + counter-reset: answer; } + .uporzadkuj .lista li { + counter-increment: answer; + padding-right: 3em; + background: url("/static/img/ornaments/draggable.c03941f77eb9.png") no-repeat 100% 50%; + } + .uporzadkuj .lista li:before { + border-radius: 0.6875em; + padding: 0.5em 0.6875em; + background-color: #d4d6d8; + color: #363a3e; + z-index: 2; + content: counter(answer); + float: left; + margin: -.5em 0 0 -3em; } + .uporzadkuj .lista li.ui-sortable-placeholder { + counter-increment: answer 0; } + .uporzadkuj .lista li.ui-sortable-helper:before { + content: none; } + +.przyporzadkuj .question-piece { + background-color: #d4d6d8; + color: #363a3e; + z-index: 2; + width: 38.625em; } +.przyporzadkuj .question-piece.short { + display: inline-block; + margin: .2em .1em; + width: auto; } +.przyporzadkuj span.question-piece { + display: inline-block; + margin: .1em; + width: auto; } +.przyporzadkuj .question-piece.disabled { + background-color: #eee; + color: #d4d6d8; } +.przyporzadkuj .predicate { + list-style: none; + margin: 0; + padding: 0; } + .przyporzadkuj .predicate > li { + border-radius: 0.6875em; + padding: 0.5em 0.6875em; + display: inline-block; + background-color: #5e6165; + color: white; + width: 11.75em; + position: relative; + vertical-align: top; } + .przyporzadkuj .predicate > li .subjects { + list-style: none; + min-height: 2.8125em; + position: relative; + margin: 0.5em -0.6875em -0.5em -0.6875em; + padding: 0.5em 0.6875em; } + .przyporzadkuj .predicate > li .subjects li { + width: auto; } + .przyporzadkuj .predicate > li .subjects .placeholder { + border-radius: 0.6875em; + padding: 0.5em 0.6875em; + text-align: right; + position: relative; + margin: 0 -0.6875em; } + .przyporzadkuj .predicate > li .subjects .multiple { + position: absolute; + z-index: -1; + top: 0; + bottom: 0; + left: 0; + right: 0; + margin: 0; } + .przyporzadkuj .predicate > li .subjects .placeholder.dragover { + z-index: 1; } + .przyporzadkuj .predicate > li .subjects .placeholder:after { + content: "upuÅÄ tutaj"; } +.przyporzadkuj .subject { + list-style: none; + margin: 0; + padding: 0; } + +.prawdafalsz .question li.question-piece { + position: relative; + padding-bottom: 0; + padding-top: 0; } +.prawdafalsz .question .question-piece-text { + display: inline-block; + width: 10cm; + vertical-align: top; + margin-top: 0.3cm; +}.prawdafalsz .question .buttons { + margin-top: 0; + top: 0.2cm; + left: 0; + position: relative; + background-color: #d4d6d8; + color: #363a3e; + display: inline-block; + margin-right: 0.5cm; } + .prawdafalsz .question .buttons a { + color: #363a3e; + text-transform: uppercase; + font-weight: bold; + display: inline-block; + padding: 0 0.6875em; } + .prawdafalsz .question .buttons a.chosen, + [data-solution="false"] .false, + [data-solution="true"] .true { + background: #ed7831; + color: white; + border: 4px solid black; + height: 29px; + box-sizing: border-box; + line-height: initial; } + [data-solution="false"] .false { + border-top-right-radius: 8px; + border-bottom-right-radius: 8px; } + [data-solution="true"] .true { + border-top-left-radius: 8px; + border-bottom-left-radius: 8px; } + +.wybor .question .lista { + list-style: none; + padding: 0; } + .wybor .question .lista li { + margin: 0; } -- 2.20.1