Changes between Initial Version and Version 1 of DeveloperGuidelinesS3Framework


Ignore:
Timestamp:
01/09/09 18:49:55 (16 years ago)
Author:
Fran Boon
Comment:

--

Legend:

Unmodified
Added
Removed
Modified
  • DeveloperGuidelinesS3Framework

    v1 v1  
     1== SahanaPy Framework ==
     2We have built an S3 framework as a higher level of abstraction on top of T2.[[BR]]
     3This should be used by modules where possible, but if more power is needed then drop down a level of two (to T2, to base Web2Py or to raw Python).
     4
     5T2 is used for:
     6 * [wiki:DeveloperGuidelinesAuthenticationAccess AAA]
     7 * simplified [wiki:DeveloperGuidelinesCreateReadUpdateDelete CRUD]
     8 * Conflict Detection
     9We extend the T2 class in {{{modules/sahana.py}}}.[[BR]]
     10Patches to the framework should be done here.[[BR]]
     11If they're generic enough then they can be suggested to Massimo for moving upstream to either Web2Py or T2.
     12
     13We extend the CRUD with a [wiki:BluePrintRESTImplementation RESTlike controller].[[BR]]
     14This provides details on how to configure your Model & Controllers. Views may not be required, other than index.html
     15
     16=== Mandatory ===
     17Each Controller should start like this (to populate the side navigation Menus):
     18{{{
     19module='module'
     20# Current Module (for sidebar title)
     21module_name=db(db.module.name==module).select()[0].name_nice
     22# List Modules (from which to build Menu of Modules)
     23modules=db(db.module.enabled=='Yes').select(db.module.ALL,orderby=db.module.menu_priority)
     24# List Options (from which to build Menu for this Module)
     25options=db(db['%s_menu_option' % module].enabled=='Yes').select(db['%s_menu_option' % module].ALL,orderby=db['%s_menu_option' % module].priority)
     26}}}
     27
     28Each function needs to return these values to the view:
     29  return dict(module_name=module_name,modules=modules,options=options)
     30
     31=== Optional ===
     32List output can be made more functional by this .represent 'widget':
     33{{{
     34def shn_list_item(table,resource,action,display='table.name',extra=None):
     35    if extra:
     36        items=DIV(TR(TD(A(eval(display),_href=t2.action(resource,[action,table.id]))),TD(eval(extra))))
     37    else:
     38        items=DIV(A(eval(display),_href=t2.action(resource,[action,table.id])))
     39    return DIV(*items)
     40}}}
     41You can use it in {{{models/module.py}}} like:
     42{{{
     43db.or_organisation.represent=lambda table:shn_list_item(table,resource='organisation',action='display')
     44db.person.represent=lambda table:shn_list_item(table,resource='person',action='display',display='table.full_name')
     45db.gis_projection.represent=lambda table:shn_list_item(table,resource='projection',action='display',extra='table.epsg')
     46}}}
     47
     48Form labels can be set in a translatable manner using:
     49   {{{db.table.field.label=T("label")}}}
     50
     51Form field can be made to use a TEXTAREA by marking the field as being type 'text':
     52   {{{SQLField('field','text'),}}}
     53
     54Form field can be made to use a SELECT dropdown by setting the field as a lookup to another table...linked to the 'uuid' field to allow [wiki:DeveloperGuidelinesDatabaseSynchronization Database Synchronization], but displaying a more user-friendly field (such as 'name'):
     55{{{
     56SQLField('field',length=64),
     57
     58db.table.field.requires=IS_NULL_OR(IS_IN_DB(db,'othertable.uuid','othertable.name'))
     59}}}
     60
     61Form field being required can be marked using:
     62   {{{db.table.field.comment=SPAN("*",_class="req")}}}
     63
     64Help for a form field can be set using:
     65   {{{A(SPAN("[Help]"),_class="popupLink",_id="tooltip",_title=T("Help Title|This is what this field is for."))}}}
     66
     67Different Flash styles can be set via:
     68{{{
     69session.error=T("Unsupported format!")
     70redirect(URL(r=request,f=resource))
     71}}}
     72or (in a Multiple Table form.accepts):
     73{{{
     74response.error=T("Form invalid!")
     75}}}
     76Supported styles are:
     77 * .warning
     78 * .error
     79 * .information
     80 * .confirmation (Standard T2 Flash messages are usually of this sort so we class them in the same way)
     81
     82=== Settings ===
     83System-wide settings have their default values set in {{{models/_db.py}}}'s {{{shn_default}}}.
     84
     85Upon 1st run of the system, these settings are pulled into the default_setting table in the database from where they can susequently be edited to configure the running instance.
     86
     87If these settings are required by all Views then can patch the session to see them in {{{models/_db.py}}}'s {{{shn_sessions()}}}.[[BR]]
     88e.g. response.debug is made available to the default {{{layout.html}}} this way.
     89
     90If you wish to clean out all settings to import afresh, then close down Web2Py & start up in shell mode:
     91{{{
     92python web2py.py -S sahana -M
     93}}}
     94Then run:
     95{{{
     96shn_db_clean(db)
     97}}}
     98(This function is configured in {{{modules/sahana.py}}})
     99
     100
     101Modules can set up their own configuration settings tables in a similar way (e.g. GIS does this)
     102
     103=== jQuery Widgets ===
     104 * [wiki:DeveloperGuidelinesDeletableList Deletable List]
     105
     106=== Conflict Detection ===
     107Sahana is a multi-user system so there is a potential for multiple users to be editing the same record at once.[[BR]]
     108We use T2 to handle this for us.
     109
     110Add this field to each table which needs protecting (in {{{models/db.py}}}):
     111{{{
     112SQLField('modified_on','datetime'), # Used by T2 to do edit conflict-detection
     113}}}
     114
     115This field is also used in [wiki:DeveloperGuidelinesDatabaseSynchronization Database Synchronization]
     116
     117----
     118DeveloperGuidelines