Changes between Initial Version and Version 1 of S3/S3Profile


Ignore:
Timestamp:
05/12/14 11:10:59 (11 years ago)
Author:
Fran Boon
Comment:

--

Legend:

Unmodified
Added
Removed
Modified
  • S3/S3Profile

    v1 v1  
     1[[TOC]]
     2= S3Profile: Profile Pages =
     3
     4A 'profile' page is designed as a way of representing a single record, along with various related resources (typically [wiki:S3/S3Model/ComponentResources Components]).
     5
     6It consists of a Header, along with a number of different Widgets:
     7* Map
     8 - displays component resources, can be made fullscreen, layers update as new records are added
     9* !DataList ('cards')
     10 - can add new records and the list will refresh accordingly
     11* !DataTable
     12 - can add new records and the list will refresh accordingly
     13* Form
     14* Report
     15* Comments (to come...)
     16
     17The profile page is accessed by calling the 'profile' method of the record:
     18* http://host.domain/eden/module/resource/record_id/profile
     19e.g.:
     20* http://host.domain/eden/pr/person/1/profile
     21
     22If there are no Widgets defined, then the profile method falls back to the simpler 'read' view for the resource (which can still include Components on Tabs if these are defined in the [wiki:DeveloperGuidelines/Tutorial/RESTCustomisation#PutitonaTab rheader])
     23
     24== Configuration ==
     25The profile pages are configured by configuring a few profile_* variables and storing these against the tablename with the usual {{{s3db.configure(tablename, key=value)}}}, e.g.:
     26{{{
     27s3db.configure(tablename,
     28               profile_title = title,
     29               profile_header = DIV(H2(title),
     30                                    _class="profile_header",
     31                                    ),
     32               profile_cols = 4
     33               profile_widgets = [alerts_widget,
     34                                  resources_widget,
     35                                  tasks_widget,
     36                                  map_widget,
     37                                  ],
     38               profile_layers = [layer],
     39               )
     40}}}
     41
     42* {{{profile_title}}} is a simple string to be used as the response.title of the page (which will show in the browser title)
     43 - if not defined, this falls-back to {{{r.record.name}}} or {{{T("Profile Page")}}}
     44* {{{profile_header}}} is HTML prepared using Web2Py's HTML helpers ({{{gluon/html.py}}}). It can be simple (as it is here) or can be as complex as you wish.
     45 - if not defined, this falls-back to {{{H2(title, _class="profile_header")}}}
     46* {{{profile_cols}}} is an integer showing how many columns the widgets are spread across
     47 - if not defined, this falls-back to 2
     48* {{{profile_widgets}}} are the Widgets (see below)
     49* {{{profile_layers}}} are additional layers to display on the map, defined like:
     50{{{
     51layer = dict(name = record_name,
     52             id = "profile-header-%s-%s" % (tablename, record_id),
     53             active = True,
     54             tablename = tablename,
     55             url = "/%s/event/incident.geojson?incident.id=%s" % \
     56                        (r.application, record_id),
     57             marker = marker,
     58             )
     59}}}
     60
     61=== Widget configuration ===
     62Here is an example configuration for a map widget:
     63{{{
     64map_widget = dict(label = "Map",
     65                  type = "map",
     66                  context = "incident",
     67                  icon = "icon-map",
     68                  height = 600,
     69                  width = 200,
     70                  colspan = 1,
     71                  )
     72}}}
     73* {{{label}}} is used as a display title of the widget
     74* {{{type}}} identifies this as a Map widget
     75* {{{context}}} is used as a filter for the map layers. See [wiki:S3/Context]
     76* {{{icon}}} is used as an icon for the widget
     77* {{{height}}} and {{{width}}} are used to size the map viewport
     78* {{{colspan}}} says how many columns are taken up by the widget
     79
     80Here is an example configuration for a datalist widget:
     81{{{
     82alerts_widget = dict(label = "Alerts",
     83                     label_create = "Create Alert",
     84                     type = "datalist",
     85                     tablename = "cms_post",
     86                     context = "incident",
     87                     # Only show Active Alerts
     88                     filter = S3FieldSelector("expired") == False,
     89                     icon = "icon-alert",
     90                     colspan = 1,
     91                     #list_layout = s3db.cms_post_list_layout,
     92                     )
     93}}}
     94Most of the options are the same as for the map widget, extra ones include:
     95* {{{label_create}}} is used as a title for the popup with the create form
     96* {{{tablename}}} is used to show which resource is used for this datalist
     97* {{{filter}}} is used as a filter for the resource (usually an additional filter on top of the Context, although it can replace the context if-desired)
     98* {{{list_layout}}} can define an alternate card renderer (falls-back to the default {{{s3db.get_config(tablename, "list_layout")}}})
     99
     100NB You should ensure that your create forms establish the context link automatically. This is usually done by checking for a context filter, and then setting table.field.default, e.g.:
     101{{{
     102incident_id = get_vars.get("~.(incident)", None)
     103if incident_id:
     104    field = table.incident_id
     105    field.default = incident_id
     106    field.readable = field.writable = False
     107}}}
     108
     109If the link is via a link table then you need to add the link onaccept, e.g.:
     110{{{
     111incident_id = r.get_vars.get("~.(incident)", None)
     112if incident_id:
     113    # Coming from Profile page
     114    # Add link onaccept
     115    def create_onaccept(form):
     116        current.s3db.event_post.insert(incident_id=incident_id,
     117                                       post_id=form.vars.id)
     118
     119    s3db.configure("cms_post",
     120                   create_onaccept = create_onaccept,
     121                   )
     122}}}
     123
     124== Code ==
     125The core functionality is in {{{modules/s3/s3profile.py}}} although this is a very small file as most of the work is in the surrounding framework of [wiki:S3/S3REST S3REST] and in the Widgets.
     126
     127The hook to link the 'profile' method to this class is in {{{s3_rest_controller()}}} in {{{models/00_utils.py}}}:
     128{{{
     129set_handler("profile", s3base.S3Profile)
     130}}}
     131
     132The default view template is in {{{views/_profile.html}}} but this can be over-ridden for a sepcific template as {{{private/templates/mytemplate/views/_profile.html}}}
     133
     134=== Widget code ===
     135==== DataList ====
     136{{{S3Profile._datalist()}}} ({{{modules/s3/s3profile.py}}}) instantiates an {{{S3DataList()}}} ({{{modules/s3/s3data.py}}}) using {{{resource.datalist()}}} and {{{datalist.html()}}}.[[BR]]
     137Client-side !JavaScript is in {{{static/scripts/S3/s3.dataTables.js}}}.
     138
     139==== DataTable ====
     140{{{S3Profile._datatable()}}} ({{{modules/s3/s3profile.py}}}) instantiates an {{{S3DataTable()}}} ({{{modules/s3/s3data.py}}}) using {{{resource.datatable()}}} and {{{datatable.html()}}}.[[BR]]
     141Client-side !JavaScript is in {{{static/scripts/S3/s3.dataLists.js}}}.
     142
     143==== Map ====
     144{{{S3Profile._map()}}} ({{{modules/s3/s3profile.py}}}) instantiates an {{{S3Map()}}} ({{{modules/s3/s3gis.py}}}) with a layer for each widget which is a datalist, datatable or report (unless that widget is configured {{{show_on_map=False}}}).[[BR]]
     145Client-side !JavaScript is in {{{static/scripts/S3/s3.gis.js}}}.
     146
     147==== Form ====
     148{{{S3Profile._form()}}} ({{{modules/s3/s3profile.py}}}) instantiates an {{{S3SQLForm()}}} ({{{modules/s3/s3forms.py}}}).[[BR]]
     149If the form has Inline components, then the client-side !JavaScript for this is in {{{static/scripts/S3/s3.inline_component.js}}}.
     150
     151==== Report ====
     152{{{S3Profile._report()}}} ({{{modules/s3/s3profile.py}}}) instantiates an {{{S3Report()}}} ({{{modules/s3/s3data.py}}}) using {{{report.widget()}}}.[[BR]]
     153Client-side !JavaScript is in {{{static/scripts/S3/s3.jquery.ui.pivottable.js}}}.
     154
     155== Future Extensions ==
     156* Allow component resources to be filtered
     157* Comments widget
     158
     159== See Also ==
     160* [wiki:S3/S3Summary Summary Pages] - the standard multi-record view