Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.
 
 
 
 
 
 

295 wiersze
8.8 KiB

  1. # Copyright (c) 2013, Web Notes Technologies Pvt. Ltd. and Contributors
  2. # MIT License. See license.txt
  3. from __future__ import unicode_literals
  4. from webnotes import conf
  5. import webnotes
  6. import json, os, time
  7. from webnotes import _
  8. import webnotes.utils
  9. import mimetypes
  10. from webnotes.website.doctype.website_sitemap.website_sitemap import add_to_sitemap
  11. class PageNotFoundError(Exception): pass
  12. def render(page_name):
  13. """render html page"""
  14. if not page_name:
  15. page_name = "index"
  16. try:
  17. html = render_page(page_name)
  18. except Exception:
  19. html = render_page("error")
  20. webnotes._response.data = html
  21. def render_page(page_name):
  22. """get page html"""
  23. set_content_type(page_name)
  24. if page_name.endswith('.html'):
  25. page_name = page_name[:-5]
  26. html = ''
  27. if not conf.disable_website_cache:
  28. html = webnotes.cache().get_value("page:" + page_name)
  29. from_cache = True
  30. if not html:
  31. html = build_page(page_name)
  32. from_cache = False
  33. if page_name=="error":
  34. html = html.replace("%(error)s", webnotes.get_traceback())
  35. elif "text/html" in webnotes._response.headers["Content-Type"]:
  36. comments = "\npage:"+page_name+\
  37. "\nload status: " + (from_cache and "cache" or "fresh")
  38. html += """\n<!-- %s -->""" % webnotes.utils.cstr(comments)
  39. return html
  40. def set_content_type(page_name):
  41. webnotes._response.headers["Content-Type"] = "text/html; charset: utf-8"
  42. if "." in page_name and not page_name.endswith(".html"):
  43. content_type, encoding = mimetypes.guess_type(page_name)
  44. webnotes._response.headers["Content-Type"] = content_type
  45. def build_page(page_name):
  46. if not webnotes.conn:
  47. webnotes.connect()
  48. if page_name=="index":
  49. page_name = get_home_page()
  50. try:
  51. sitemap_options = webnotes.doc("Website Sitemap", page_name).fields
  52. page_options = webnotes.doc("Website Sitemap Config",
  53. sitemap_options.get("website_sitemap_config")).fields.update({
  54. "page_name":sitemap_options.page_name,
  55. "docname":sitemap_options.docname
  56. })
  57. except webnotes.DoesNotExistError:
  58. hooks = webnotes.get_hooks()
  59. if hooks.website_catch_all:
  60. return build_page(hooks.website_catch_all[0])
  61. page_options["page_name"] = page_name
  62. no_cache = page_options.get("no_cache")
  63. # if generator, then load bean, pass arguments
  64. if page_options.get("page_or_generator")=="Generator":
  65. bean = webnotes.bean(page_options.get("ref_doctype"), page_options["docname"])
  66. bean.run_method("get_context")
  67. context = webnotes._dict(bean.doc.fields)
  68. context["obj"] = bean.get_controller()
  69. else:
  70. # page
  71. context = webnotes._dict({ 'name': page_name })
  72. if page_options.get("controller"):
  73. module = webnotes.get_module(page_options.get("controller"))
  74. if module and hasattr(module, "get_context"):
  75. context.update(module.get_context())
  76. context.update(get_website_settings())
  77. jenv = webnotes.get_jenv()
  78. context["base_template"] = jenv.get_template("templates/base.html")
  79. template_name = page_options['template_path']
  80. context["_"] = webnotes._
  81. html = jenv.get_template(template_name).render(context)
  82. if not no_cache:
  83. webnotes.cache().set_value("page:" + page_name, html)
  84. return html
  85. def get_home_page():
  86. return webnotes.cache().get_value("home_page", \
  87. lambda: webnotes.conn.get_value("Website Settings", None, "home_page") or "login")
  88. def get_website_settings():
  89. from webnotes.utils import get_request_site_address, encode, cint
  90. from urllib import quote
  91. hooks = webnotes.get_hooks()
  92. all_top_items = webnotes.conn.sql("""\
  93. select * from `tabTop Bar Item`
  94. where parent='Website Settings' and parentfield='top_bar_items'
  95. order by idx asc""", as_dict=1)
  96. top_items = [d for d in all_top_items if not d['parent_label']]
  97. # attach child items to top bar
  98. for d in all_top_items:
  99. if d['parent_label']:
  100. for t in top_items:
  101. if t['label']==d['parent_label']:
  102. if not 'child_items' in t:
  103. t['child_items'] = []
  104. t['child_items'].append(d)
  105. break
  106. context = webnotes._dict({
  107. 'top_bar_items': top_items,
  108. 'footer_items': webnotes.conn.sql("""\
  109. select * from `tabTop Bar Item`
  110. where parent='Website Settings' and parentfield='footer_items'
  111. order by idx asc""", as_dict=1),
  112. "webnotes": webnotes,
  113. "utils": webnotes.utils,
  114. "post_login": [
  115. {"label": "Reset Password", "url": "update-password", "icon": "icon-key"},
  116. {"label": "Logout", "url": "/?cmd=web_logout", "icon": "icon-signout"}
  117. ]
  118. })
  119. settings = webnotes.doc("Website Settings", "Website Settings")
  120. for k in ["banner_html", "brand_html", "copyright", "twitter_share_via",
  121. "favicon", "facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  122. "disable_signup"]:
  123. if k in settings.fields:
  124. context[k] = settings.fields.get(k)
  125. if settings.address:
  126. context["footer_address"] = settings.address
  127. for k in ["facebook_share", "google_plus_one", "twitter_share", "linked_in_share",
  128. "disable_signup"]:
  129. context[k] = cint(context.get(k) or 0)
  130. context.url = quote(str(get_request_site_address(full_address=True)), str(""))
  131. context.encoded_title = quote(encode(context.title or ""), str(""))
  132. for update_website_context in hooks.update_website_context or []:
  133. webnotes.get_attr(update_website_context)(context)
  134. context.web_include_js = hooks.web_include_js or []
  135. context.web_include_css = hooks.web_include_css or []
  136. return context
  137. def clear_cache(page_name=None):
  138. if page_name:
  139. delete_page_cache(page_name)
  140. else:
  141. cache = webnotes.cache()
  142. for p in webnotes.conn.sql_list("""select name from `tabWebsite Sitemap`"""):
  143. if p is not None:
  144. cache.delete_value("page:" + p)
  145. cache.delete_value("home_page")
  146. cache.delete_value("page:index")
  147. cache.delete_value("website_sitemap")
  148. cache.delete_value("website_sitemap_config")
  149. def delete_page_cache(page_name):
  150. if page_name:
  151. cache = webnotes.cache()
  152. cache.delete_value("page:" + page_name)
  153. cache.delete_value("website_sitemap")
  154. def is_signup_enabled():
  155. if getattr(webnotes.local, "is_signup_enabled", None) is None:
  156. webnotes.local.is_signup_enabled = True
  157. if webnotes.utils.cint(webnotes.conn.get_value("Website Settings",
  158. "Website Settings", "disable_signup")):
  159. webnotes.local.is_signup_enabled = False
  160. return webnotes.local.is_signup_enabled
  161. class WebsiteGenerator(object):
  162. def setup_generator(self):
  163. if webnotes.flags.in_install_app:
  164. return
  165. self._website_config = webnotes.conn.get_values("Website Sitemap Config",
  166. {"ref_doctype": self.doc.doctype}, "*")[0]
  167. def on_update(self, page_name=None):
  168. if webnotes.flags.in_install_app:
  169. return
  170. self.setup_generator()
  171. if self._website_config.condition_field:
  172. if not self.doc.fields.get(self._website_config.condition_field):
  173. # condition field failed, return
  174. remove_page(self.doc.fields.get(self._website_config.page_name_field))
  175. return
  176. if not page_name:
  177. new_page_name = cleanup_page_name(self.get_page_title() \
  178. if hasattr(self, "get_page_title") else (self.doc.title or self.doc.name))
  179. self.check_if_page_name_is_unique(new_page_name)
  180. remove_page(self.doc.page_name)
  181. if self.doc.fields.get(self._website_config.page_name_field)!=new_page_name:
  182. webnotes.conn.set(self.doc, self._website_config.page_name_field, new_page_name)
  183. page_name = new_page_name
  184. add_to_sitemap(webnotes._dict({
  185. "ref_doctype":self.doc.doctype,
  186. "docname": self.doc.name,
  187. "page_name": page_name,
  188. "link_name": self._website_config.name,
  189. "lastmod": webnotes.utils.get_datetime(self.doc.modified).strftime("%Y-%m-%d")
  190. }))
  191. def check_if_page_name_is_unique(self, new_page_name):
  192. if webnotes.conn.sql("""select name from `tabWebsite Sitemap` where name=%s
  193. and website_sitemap_config!=%s and docname!=%s""", (new_page_name,
  194. self._website_config.name, self.doc.name)):
  195. webnotes.throw("%s: %s. %s: <b>%s<b>" % (new_page_name, _("Page already exists"),
  196. _("Please change the value"), self.doc.title))
  197. def on_trash(self):
  198. self.setup_generator()
  199. remove_page(self.doc.fields[self._website_config.page_name_field])
  200. def remove_page(page_name):
  201. if page_name:
  202. delete_page_cache(page_name)
  203. webnotes.conn.sql("delete from `tabWebsite Sitemap` where name=%s", page_name)
  204. def cleanup_page_name(title):
  205. """make page name from title"""
  206. import re
  207. name = title.lower()
  208. name = re.sub('[~!@#$%^&*+()<>,."\'\?]', '', name)
  209. name = re.sub('[:/]', '-', name)
  210. name = '-'.join(name.split())
  211. # replace repeating hyphens
  212. name = re.sub(r"(-)\1+", r"\1", name)
  213. return name
  214. def get_hex_shade(color, percent):
  215. def p(c):
  216. v = int(c, 16) + int(int('ff', 16) * (float(percent)/100))
  217. if v < 0:
  218. v=0
  219. if v > 255:
  220. v=255
  221. h = hex(v)[2:]
  222. if len(h) < 2:
  223. h = "0" + h
  224. return h
  225. r, g, b = color[0:2], color[2:4], color[4:6]
  226. avg = (float(int(r, 16) + int(g, 16) + int(b, 16)) / 3)
  227. # switch dark and light shades
  228. if avg > 128:
  229. percent = -percent
  230. # stronger diff for darker shades
  231. if percent < 25 and avg < 64:
  232. percent = percent * 2
  233. return p(r) + p(g) + p(b)