+++ /dev/null
-#####
-### Epoz - a cross-browser-wysiwyg-editor for Zope
-## Copyright (C) 2005 Maik Jablonski (maik.jablonski@uni-bielefeld.de)
-#
-# All Rights Reserved.
-#
-# This software is subject to the provisions of the Zope Public License,
-# Version 2.1 (ZPL). A copy of the ZPL should accompany this distribution.
-# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
-# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
-## WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
-### FOR A PARTICULAR PURPOSE.
-####
-
-import re
-import urlparse
-import cgi
-
-# try to import mxTidy as default
-try:
- from mx import Tidy
- mxTidyIsAvailable = 1
-except:
- mxTidyIsAvailable = 0
-
-# try to import uTidylib as fallback
-try:
- import tidy
- uTidyIsAvailable = 1
-except ImportError:
- uTidyIsAvailable = 0
-
-import Globals
-from Products.PageTemplates.PageTemplateFile import PageTemplateFile
-from App.ImageFile import ImageFile
-
-# Regexp for snipping the body of a document
-HTML_BODY = re.compile('<body[^>]*?>(.*)</body[^>]*?>', re.DOTALL|re.IGNORECASE)
-MSO_CLASS = re.compile(r'<(\w+)\W+(class="Mso.*?")>', re.IGNORECASE)
-EPOZ_SCRIPT = re.compile(r'<epoz:script (style="[^"]*")? ?')
-
-misc_ = {
- # The Format-Controls
-'epoz_script_main.js':ImageFile('epoz/epoz_core/epoz_script_main.js.dtml', globals()),
-
- # Rendering of the widget
-'epoz_script_widget.js':ImageFile('epoz/epoz_core/epoz_script_widget.js.dtml', globals()),
-
- # Browser-detection
-'epoz_script_detect.js':ImageFile('epoz/epoz_core/epoz_script_detect.js.dtml', globals()),
-
- # Code for the color-selection-popup
-'epoz_script_color.html':ImageFile('epoz/epoz_core/epoz_script_color.html.dtml', globals()),
-
- # Code for the table-popup
-'epoz_script_table.html':ImageFile('epoz/epoz_core/epoz_script_table.html.dtml', globals()),
-
- # The XMLRPC-handler
-'vcXMLRPC.js':ImageFile('epoz/epoz_core/vcXMLRPC.js.dtml', globals()),
-
-# Multi-Epoz
-'epoz_redirect.js':ImageFile('epoz/epoz_core/epoz_redirect.js.dtml', globals()),
-'epoz_iframe_trigger.js':ImageFile('epoz/epoz_core/epoz_iframe_trigger.js.dtml', globals()),
-
- # The Epoz-Language-Files
-'epoz_lang_da.js':ImageFile('epoz/epoz_i18n/epoz_lang_da.js.dtml', globals()),
-'epoz_lang_de.js':ImageFile('epoz/epoz_i18n/epoz_lang_de.js.dtml', globals()),
-'epoz_lang_en.js':ImageFile('epoz/epoz_i18n/epoz_lang_en.js.dtml', globals()),
-'epoz_lang_es.js':ImageFile('epoz/epoz_i18n/epoz_lang_es.js.dtml', globals()),
-'epoz_lang_fi.js':ImageFile('epoz/epoz_i18n/epoz_lang_fi.js.dtml', globals()),
-'epoz_lang_fr.js':ImageFile('epoz/epoz_i18n/epoz_lang_fr.js.dtml', globals()),
-'epoz_lang_hu.js':ImageFile('epoz/epoz_i18n/epoz_lang_hu.js.dtml', globals()),
-'epoz_lang_it.js':ImageFile('epoz/epoz_i18n/epoz_lang_it.js.dtml', globals()),
-'epoz_lang_nl.js':ImageFile('epoz/epoz_i18n/epoz_lang_nl.js.dtml', globals()),
-'epoz_lang_no.js':ImageFile('epoz/epoz_i18n/epoz_lang_no.js.dtml', globals()),
-'epoz_lang_pl.js':ImageFile('epoz/epoz_i18n/epoz_lang_pl.js.dtml', globals()),
-'epoz_lang_pt.js':ImageFile('epoz/epoz_i18n/epoz_lang_pt.js.dtml', globals()),
-'epoz_lang_pt-br.js':ImageFile('epoz/epoz_i18n/epoz_lang_pt-br.js.dtml', globals()),
-'epoz_lang_ru.js':ImageFile('epoz/epoz_i18n/epoz_lang_ru.js.dtml', globals()),
-'epoz_lang_zh-cn.js':ImageFile('epoz/epoz_i18n/epoz_lang_zh-cn.js.dtml', globals()),
-
-# Generic language file transalted by PlacelessTranslationService (for example)
-'epoz_multilingual.js':PageTemplateFile('epoz/epoz_core/epoz_multilingual.js.pt', globals()),
-
- # The Epoz-Buttons
-'epoz_button_anchor.gif':ImageFile('epoz/epoz_images/epoz_button_anchor.gif', globals()),
-'epoz_button_bgcolor.gif':ImageFile('epoz/epoz_images/epoz_button_bgcolor.gif', globals()),
-'epoz_button_bold.gif':ImageFile('epoz/epoz_images/epoz_button_bold.gif', globals()),
-'epoz_button_centre.gif':ImageFile('epoz/epoz_images/epoz_button_centre.gif', globals()),
-'epoz_button_hr.gif':ImageFile('epoz/epoz_images/epoz_button_hr.gif', globals()),
-'epoz_button_hyperlink.gif':ImageFile('epoz/epoz_images/epoz_button_hyperlink.gif', globals()),
-'epoz_button_image.gif':ImageFile('epoz/epoz_images/epoz_button_image.gif', globals()),
-'epoz_button_indent.gif':ImageFile('epoz/epoz_images/epoz_button_indent.gif', globals()),
-'epoz_button_italic.gif':ImageFile('epoz/epoz_images/epoz_button_italic.gif', globals()),
-'epoz_button_left_just.gif':ImageFile('epoz/epoz_images/epoz_button_left_just.gif', globals()),
-'epoz_button_list.gif':ImageFile('epoz/epoz_images/epoz_button_list.gif', globals()),
-'epoz_button_numbered_list.gif':ImageFile('epoz/epoz_images/epoz_button_numbered_list.gif', globals()),
-'epoz_button_outdent.gif':ImageFile('epoz/epoz_images/epoz_button_outdent.gif', globals()),
-'epoz_button_redo.gif':ImageFile('epoz/epoz_images/epoz_button_redo.gif', globals()),
-'epoz_button_right_just.gif':ImageFile('epoz/epoz_images/epoz_button_right_just.gif', globals()),
-'epoz_button_strikethrough.gif':ImageFile('epoz/epoz_images/epoz_button_strikethrough.gif', globals()),
-'epoz_button_subscript.gif':ImageFile('epoz/epoz_images/epoz_button_subscript.gif', globals()),
-'epoz_button_superscript.gif':ImageFile('epoz/epoz_images/epoz_button_superscript.gif', globals()),
-'epoz_button_table.gif':ImageFile('epoz/epoz_images/epoz_button_table.gif', globals()),
-'epoz_button_textcolor.gif':ImageFile('epoz/epoz_images/epoz_button_textcolor.gif', globals()),
-'epoz_button_tools.gif':ImageFile('epoz/epoz_images/epoz_button_tools.gif', globals()),
-'epoz_button_underline.gif':ImageFile('epoz/epoz_images/epoz_button_underline.gif', globals()),
-'epoz_button_undo.gif':ImageFile('epoz/epoz_images/epoz_button_undo.gif', globals()),
-'epoz_button_unformat.gif':ImageFile('epoz/epoz_images/epoz_button_unformat.gif', globals()),
-}
-
-
-def Epoz(self, name, data='', toolbox='', lang='auto',
- path='', widget='',
- style='width: 600px; height: 250px; border: 1px solid #000000;',
- button='background-color: #EFEFEF; border: 1px solid #A0A0A0; cursor: pointer; margin-right: 1px; margin-bottom: 1px;',
- css='', customcss='', charset='utf-8', pageurl=''):
- """ Create an Epoz-Wysiwyg-Editor.
-
- name : the name of the form-element which submits the data
- data : the data to edit
- toolbox : a link to a HTML-Page which delivers additional tools
- lang: a code for the language-file (en,de,...).
- Use auto to enable browser language detection.
- In this case, a localization product must be installed (like PlacelessTranslationService).
- path : path to Epoz-Javascript. Needed mainly for Plone (portal_url).
- widget: You can specify a path to an alternative JavaScript for
- epoz_script_widget.js
- style : style-definition for the editor-area
- button : style-definiton for buttons
- css : url to a global css which should be used for rendering
- content in editor-area
- customcss: url to a customized css which should be used for rendering
- content in editor-area
- charset : charset which is edited in the editor-area
- pageurl: the base-url for the edited page
-
- If Epoz can't create a Rich-Text-Editor, a simple textarea is created.
- """
-
- if data is None:
- data=''
-
- js_data = data
- data = cgi.escape(data)
-
- # hide scripts without removing them
- js_data = js_data.replace('<script ',
- '<epoz:script style="display: none" ')
- js_data = js_data.replace('</script>', '</epoz:script>')
-
- # Quote newlines and single quotes, so the Epoz-JavaScript won't break.
- # Needs to be a list and no dictionary, cause we need order!!!
-
- quotes = (("\\","\\\\"), ("\n","\\n"), ("\r","\\r"), ("'","\\'"))
-
- for item in quotes:
- js_data = js_data.replace(item[0], item[1])
-
- # Determine the correct path for VirtualHostMonsters & PCGI & etc.pp.
- if not path:
- path = "%s/misc_/Epoz/" % (self.REQUEST.get('BASEPATH1',''),)
-
- if not pageurl:
- pageurl = self.REQUEST.get('URL1','')
-
- if lang == 'auto' :
- i18n='<script language="JavaScript" type="text/javascript" src="%(path)sepoz_multilingual.js?charset=%(charset)s" charset="%(charset)s"></script>' % {'path' : path, 'charset' : charset}
- else :
- i18n='<script language="JavaScript" type="text/javascript" src="%sepoz_lang_en.js"></script>' % (path,)
- if lang<>'en':
- i18n += '<script language="JavaScript" type="text/javascript" src="%sepoz_lang_%s.js"></script>' % (path, lang)
-
- if not widget:
- widget = "%sepoz_script_widget.js" % path
-
- # This is just a dummy page for test-iframe
- # to prevent IE complaining when using Epoz via https
- iframesrc = pageurl+'/epoz_blank_iframe.html'
-
- # Return the HTML-Code for the Epoz-Rich-Text-Editor
- return """
-%(i18n)s
-<script language="JavaScript" type="text/javascript" src="%(widget)s"></script>
-<script language="JavaScript" type="text/javascript" src="%(path)svcXMLRPC.js"></script>
-<script language="JavaScript" type="text/javascript" src="%(path)sepoz_script_detect.js"></script>
-<script language="JavaScript" type="text/javascript" src="%(path)sepoz_script_main.js" charset="utf-8"></script>
-<script language="JavaScript" type="text/javascript" src="%(path)sepoz_redirect.js"></script>
-<script language="JavaScript" type="text/javascript">
-<!--
- InitIframe('%(name)s','%(js_data)s','%(path)s','%(toolbox)s','%(style)s','%(button)s','%(css)s','%(customcss)s','%(charset)s', '%(pageurl)s');
-//-->
-</script>
-<noscript><textarea name="%(name)s" style="%(style)s">%(data)s</textarea></noscript>
-""" % {
- 'i18n': i18n,
- 'widget': widget,
- 'path': path,
- 'name': name,
- 'js_data': js_data,
- 'toolbox': toolbox,
- 'style': style,
- 'button': button,
- 'css': css,
- 'customcss': customcss,
- 'charset': charset,
- 'data': data,
- 'pageurl': pageurl,
- 'iframesrc': iframesrc
- }
-
-
-
-def EpozTidy(self, html, pageurl):
- """ Take html and deliver xhtml if mxTidy is installed;
- call EpozPostTidy for html-postprocessings before returning the result
- """
-
- errors = 0
- output = html
- errordata = ""
-
- input = html.encode("utf-8")
- input = EPOZ_SCRIPT.sub('<script ', input)
- input = input.replace('</epoz:script>', '</script>')
-
- if mxTidyIsAvailable:
- (errors, warnings, output, errordata) = Tidy.tidy(
- input, drop_empty_paras=1, indent_spaces=1, indent="auto",
- output_xhtml=1, word_2000=1, wrap=79, char_encoding="utf8")
- if errors:
- output = html
- elif uTidyIsAvailable:
- parsed = tidy.parseString(
- input, drop_empty_paras=1, indent_spaces=1, indent="auto",
- output_xhtml=1, word_2000=1, wrap=79, char_encoding="utf8",
- add_xml_decl=0, doctype="omit", indent_attributes=1,
- drop_proprietary_attributes=1, bare=1, clean=1,
- enclose_text=1, tidy_mark=0)
- reports = parsed.get_errors()
- all_errors = [str(x) for x in reports if x.severity != 'W']
- errors = len(all_errors)
- errordata = '\n'.join(all_errors)
- if errors:
- output = html
- else:
- output = str(parsed)
-
- output = MSO_CLASS.sub(r"<\1>", output)
- result = HTML_BODY.search(output)
- if result:
- output = result.group(1)
-
- # Call External Method / PythonScript for postprocessing
- # The script should expect two parameters:
- # self = called context (=server)
- # html = the htmlbody to postprocess
- # pathname = path of edited object (maybe with template!)
- # The script should return the new htmlbody
-
- EpozPostTidy = getattr(self, 'EpozPostTidy', None)
- if EpozPostTidy is not None:
- output = EpozPostTidy(self, output, pageurl)
-
- return (errors, output, errordata)
-
-
-def EpozGetRelativeUrl(self, pageUrl, targetUrl):
- """ Returns relative path from targetUrl to pageUrl. """
-
- # Just some shortcuts...
- SCHEME, HOSTNAME, PATH, PARAMS, QUERY, FRAGMENT = range(6)
-
- pageUnits = urlparse.urlparse(pageUrl)
- targetUnits = urlparse.urlparse(targetUrl)
-
- if pageUnits[:PATH] != targetUnits[:PATH]:
- # Scheme (http) or host:port (www.comain.com) are different
- # -> no possible relative URL
- return targetUrl
-
- afterPath = targetUnits[PARAMS:]
-
- pagePath = pageUnits[PATH].split('/')
- pagePath.reverse()
-
- targetPath = targetUnits[PATH].split('/')
- targetPath.reverse()
-
- # Remove parts which are equal (['a', 'b'] ['a', 'c'] --> ['c'])
- while (len(pagePath) > 0
- and len(targetPath) > 0
- and pagePath[-1] == targetPath[-1]):
- pagePath.pop()
- last_pop = targetPath.pop()
- if len(pagePath) == 0 and len(targetPath) == 0:
- # Special case: link to itself (http://foo/a http://foo/a -> a)
- targetPath.append(last_pop)
- if len(pagePath) == 1 and pagePath[0] == "" and len(targetPath) == 0:
- # Special case: (http://foo/a/ http://foo/a -> ../a)
- pagePath.append(last_pop)
- targetPath.append(last_pop)
-
- targetPath.reverse()
-
- relativeUrl = ['..'] * (len(pagePath) -1)
- relativeUrl = ('', '', '/'.join(relativeUrl + targetPath)) + tuple(afterPath)
-
- return urlparse.urlunparse(relativeUrl)
-
-
-# Make sure the Epoz method is replaceable in the Products folder
-Epoz.__replaceable__ = Globals.REPLACEABLE
-
-# Register Epoz as global method
-# Register EpozTidy as global method for XMLRPC
-# Register EpozGetRelativeUrl as global method
-# Register epoz_blank_iframe.html as global PageTemplate
-
-methods = {
-'Epoz': Epoz,
-'Epoz__roles__': None,
-
-'EpozTidy': EpozTidy,
-'EpozTidy__roles__': None,
-
-'EpozGetRelativeUrl': EpozGetRelativeUrl,
-'EpozGetRelativeUrl__roles__': None,
-
-'epoz_blank_iframe.html': PageTemplateFile('epoz/epoz_core/epoz_blank_iframe.html.pt', globals()),
-'epoz_blank_iframe.html__roles__': None,
-}
-
-
-# And now try to register Epoz for CMF/Plone
-
-try:
- from Products.CMFCore.DirectoryView import registerDirectory
-
- global cmfepoz_globals
- cmfepoz_globals=globals()
-
- def initialize(context):
- registerDirectory('epoz', cmfepoz_globals)
-except:
- pass
-
\ No newline at end of file