Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 # Use of this source code is governed by a BSD-style license that can be | 2 # Use of this source code is governed by a BSD-style license that can be |
| 3 # found in the LICENSE file. | 3 # found in the LICENSE file. |
| 4 | 4 |
| 5 import copy | 5 import copy |
| 6 import json | |
| 7 import logging | 6 import logging |
| 8 | 7 |
| 9 class SidenavDataSource(object): | 8 from data_source import DataSource |
| 10 """This class reads in and caches a JSON file representing the side navigation | 9 from third_party.json_schema_compiler.json_parse import Parse |
| 11 menu. | |
| 12 """ | |
| 13 class Factory(object): | |
| 14 def __init__(self, compiled_fs_factory, json_path): | |
| 15 self._cache = compiled_fs_factory.Create(self._CreateSidenavDict, | |
| 16 SidenavDataSource) | |
| 17 self._json_path = json_path | |
| 18 | 10 |
| 19 def Create(self, path): | 11 _JSON_FILE_NAME = '%s/%s_sidenav.json' |
|
Jeffrey Yasskin
2013/08/28 19:12:37
This is only used in one place, so it might be bet
jshumway
2013/08/29 00:13:09
Done.
| |
| 20 """Create a SidenavDataSource, binding it to |path|. |path| is the url | |
| 21 of the page that is being rendered. It is used to determine which item | |
| 22 in the sidenav should be highlighted. | |
| 23 """ | |
| 24 return SidenavDataSource(self._cache, self._json_path, path) | |
| 25 | 12 |
| 26 def _AddLevels(self, items, level): | |
| 27 """Levels represent how deeply this item is nested in the sidenav. We | |
| 28 start at 2 because the top <ul> is the only level 1 element. | |
| 29 """ | |
| 30 for item in items: | |
| 31 item['level'] = level | |
| 32 if 'items' in item: | |
| 33 self._AddLevels(item['items'], level + 1) | |
| 34 | 13 |
| 35 def _CreateSidenavDict(self, json_path, json_str): | 14 def _AddLevels(items, level): |
| 36 items = json.loads(json_str) | 15 '''Add a 'level' key to each item in |items|. 'level' corresponds to how deep |
|
Jeffrey Yasskin
2013/08/28 19:12:37
Thanks for adding comments.
| |
| 37 self._AddLevels(items, 2); | 16 in |items| an item is. |level| sets the starting depth. |
| 38 return items | 17 ''' |
| 18 for item in items: | |
| 19 item['level'] = level | |
| 20 if 'items' in item: | |
| 21 _AddLevels(item['items'], level + 1) | |
| 39 | 22 |
| 40 def __init__(self, cache, json_path, path): | |
| 41 self._cache = cache | |
| 42 self._json_path = json_path | |
| 43 self._href = '/' + path | |
| 44 | 23 |
| 45 def _AddSelected(self, items): | 24 def _AddSelected(items, path): |
| 46 for item in items: | 25 '''Add 'selected' and 'child_selected' properties to |items| so that the |
| 47 if item.get('href', '') == self._href: | 26 sidenav can be expanded to show which menu item has been selected. |
|
Jeffrey Yasskin
2013/08/28 19:12:37
Mention what this function returns.
jshumway
2013/08/29 00:13:09
Done.
| |
| 48 item['selected'] = True | 27 ''' |
| 28 for item in items: | |
| 29 if item.get('href', '') == '/' + path: | |
| 30 item['selected'] = True | |
| 31 return True | |
| 32 if 'items' in item: | |
| 33 if _AddSelected(item['items'], path): | |
| 34 item['child_selected'] = True | |
| 49 return True | 35 return True |
| 50 if 'items' in item: | |
| 51 if self._AddSelected(item['items']): | |
| 52 item['child_selected'] = True | |
| 53 return True | |
| 54 return False | |
| 55 | 36 |
| 56 def _QualifyHrefs(self, items): | 37 return False |
| 57 for item in items: | |
| 58 if 'items' in item: | |
| 59 self._QualifyHrefs(item['items']) | |
| 60 | 38 |
| 61 href = item.get('href') | 39 |
| 62 if href is not None and not href.startswith(('http://', 'https://')): | 40 def _QualifyHrefs(items): |
| 63 if not href.startswith('/'): | 41 '''Force hrefs in |items| to either be absolute (http://...) or qualified |
| 64 logging.warn('Paths in sidenav must be qualified. %s is not.' % href) | 42 (begins with a slash (/)). Other hrefs emit a warning and should be updated. |
| 65 href = '/' + href | 43 ''' |
| 66 item['href'] = href | 44 for item in items: |
| 45 if 'items' in item: | |
| 46 _QualifyHrefs(item['items']) | |
| 47 | |
| 48 href = item.get('href') | |
| 49 if href is not None and not href.startswith(('http://', 'https://')): | |
| 50 if not href.startswith('/'): | |
| 51 logging.warn('Paths in sidenav must be qualified. %s is not.' % href) | |
| 52 href = '/' + href | |
| 53 item['href'] = href | |
| 54 | |
| 55 | |
| 56 class SidenavDataSource(DataSource): | |
| 57 '''Provides templates with access to JSON files used to create the side | |
| 58 navigation bar. | |
| 59 ''' | |
| 60 def __init__(self, server_instance, request): | |
| 61 self._cache = server_instance.compiled_host_fs_factory.Create( | |
| 62 self._CreateSidenavDict, SidenavDataSource) | |
| 63 self._json_path = server_instance.sidenav_json_base_path | |
| 64 self._request = request | |
| 65 | |
| 66 def _CreateSidenavDict(self, _, content): | |
|
Jeffrey Yasskin
2013/08/28 19:12:37
It doesn't look like you use self, so can you make
jshumway
2013/08/29 00:13:09
Done.
| |
| 67 items = Parse(content) | |
| 68 # Start at level 2, the top <ul> element is level 1. | |
| 69 _AddLevels(items, level=2) | |
| 70 _QualifyHrefs(items) | |
| 71 return items | |
| 72 | |
| 73 def Cron(self): | |
| 74 for platform in ['apps', 'extensions']: | |
| 75 self._cache.GetFromFile(_JSON_FILE_NAME % (self._json_path, platform)) | |
| 67 | 76 |
| 68 def get(self, key): | 77 def get(self, key): |
| 69 sidenav = copy.deepcopy(self._cache.GetFromFile( | 78 sidenav = copy.deepcopy(self._cache.GetFromFile( |
| 70 '%s/%s_sidenav.json' % (self._json_path, key))) | 79 '%s/%s_sidenav.json' % (self._json_path, key))) |
| 71 self._AddSelected(sidenav) | 80 _AddSelected(sidenav, self._request.path) |
| 72 self._QualifyHrefs(sidenav) | |
| 73 return sidenav | 81 return sidenav |
| OLD | NEW |