From dd102ec0c6984b6247fe169523fd00ce17104a40 Mon Sep 17 00:00:00 2001 From: Romain Courteaud <romain@nexedi.com> Date: Fri, 24 Mar 2006 16:47:11 +0000 Subject: [PATCH] Unify RelationField and MultiRelationField (in order to fix both of them at the same time). Bug fix: do not write the relation_index in the ZODB anymore. git-svn-id: https://svn.erp5.org/repos/public/erp5/trunk@6202 20353a03-c40f-0410-a6d1-a30d3c3de9de --- product/ERP5Form/Form.py | 9 +- product/ERP5Form/MultiRelationField.py | 1052 +++++++++++++++--------- product/ERP5Form/RelationField.py | 529 +++--------- product/ERP5Form/SelectionTool.py | 265 +++--- 4 files changed, 854 insertions(+), 1001 deletions(-) diff --git a/product/ERP5Form/Form.py b/product/ERP5Form/Form.py index 413a77de79..83bc81f505 100755 --- a/product/ERP5Form/Form.py +++ b/product/ERP5Form/Form.py @@ -38,7 +38,6 @@ from urllib import quote from Globals import InitializeClass, PersistentMapping, DTMLFile, get_request from AccessControl import Unauthorized, getSecurityManager, ClassSecurityInfo from ZODB.POSException import ConflictError - from Products.ERP5Type.Utils import UpperCase import psyco @@ -111,6 +110,7 @@ def get_value(self, id, **kw): else: # get normal value value = self.get_orig_value(id) + # For the 'default' value, we try to get a default value if id == 'default': if (value is None or value == '' or value == [] or value == ()) \ @@ -355,7 +355,6 @@ class ERP5Form(ZMIForm, ZopePageTemplate): # Proxy method to PageTemplate def __call__(self, *args, **kwargs): - self._v_relation_field_index = 0 # We initialize here an index which is used to generate different method ids for every field if not kwargs.has_key('args'): kwargs['args'] = args form = self @@ -370,6 +369,11 @@ class ERP5Form(ZMIForm, ZopePageTemplate): extra_context['form'] = self extra_context['container'] = container ## PROBLEM NOT TAKEN INTO ACCOUNT extra_context['here'] = object + # We initialize here an index which is used to generate + # different method ids for every field + request = extra_context['request'] + # XXX We must not use a counter, but a ID for each field + request.set('_v_relation_field_index', 0) return pt.pt_render(extra_context=extra_context) def _exec(self, bound_names, args, kw): @@ -456,6 +460,7 @@ class ERP5Form(ZMIForm, ZopePageTemplate): # More optimizations #psyco.bind(ERP5Field) +# XXX Not useful, as we patch those methods in FormulatorPatch psyco.bind(Field.render) psyco.bind(Field._render_helper) psyco.bind(Field.get_value) diff --git a/product/ERP5Form/MultiRelationField.py b/product/ERP5Form/MultiRelationField.py index c8286224f6..c4d2bcbdc0 100755 --- a/product/ERP5Form/MultiRelationField.py +++ b/product/ERP5Form/MultiRelationField.py @@ -1,6 +1,7 @@ ############################################################################## # -# Copyright (c) 2002, 2004 Nexedi SARL and Contributors. All Rights Reserved. +# Copyright (c) 2002, 2004, 2006 Nexedi SARL and Contributors. +# All Rights Reserved. # Jean-Paul Smets-Solanes <jp@nexedi.com> # Romain Courteaud <romain@nexedi.com> # @@ -29,19 +30,22 @@ from Products.Formulator import Widget, Validator from Products.Formulator.Field import ZMIField -from Products.Formulator.DummyField import fields from Products.ERP5Type.Utils import convertToUpperCase from Products.CMFCore.utils import getToolByName -from Products.ERP5Form import RelationField -from Products.ERP5Form.RelationField import MAX_SELECT, new_content_prefix -from Globals import get_request from Products.PythonScripts.Utility import allow_class +from Products.ERP5Type.Message import Message from AccessControl import ClassSecurityInfo - -import string +from types import StringType from zLOG import LOG -#MAX_SELECT = 50 # Max. number of catalog result -#new_content_prefix = '_newContent_' +from Products.Formulator.DummyField import fields +from Globals import get_request + +# Max. number of catalog result +MAX_SELECT = 30 +NEW_CONTENT_PREFIX = '_newContent_' +# Key for sub listfield +SUB_FIELD_ID = 'relation' +ITEM_ID = 'item' def checkSameKeys(a , b): """ @@ -52,199 +56,352 @@ def checkSameKeys(a , b): for ka in a: if (not ka in b) and (ka != ''): same = 0 - for kb in b: - if (not kb in a) and (kb != ''): - same = 0 + break + if same: + for kb in b: + if (not kb in a) and (kb != ''): + same = 0 + break return same - -class MultiRelationStringFieldWidget(Widget.LinesTextAreaWidget, RelationField.RelationStringFieldWidget): +class MultiRelationStringFieldWidget(Widget.LinesTextAreaWidget, + Widget.TextWidget, + Widget.ListWidget): + """ + RelationStringField widget + Works like a string field but includes one buttons + - one search button which updates the field and sets a relation + - creates object if not there + """ + local_property_names = ['update_method', 'jump_method', 'allow_jump', + 'base_category', 'portal_type', 'allow_creation', + 'container_getter_id', 'catalog_index', + 'relation_setter_id', 'columns', 'sort', + 'parameter_list','list_method', + 'first_item', 'items', 'size', 'extra_item'] + + property_names = Widget.LinesTextAreaWidget.property_names + \ + Widget.TextWidget.property_names + \ + local_property_names + + # XXX Field to remove... + update_method = fields.StringField('update_method', + title='Update Method', + description=( + "The method to call to set the relation. Required."), + default="Base_validateRelation", + required=1) + + jump_method = fields.StringField('jump_method', + title='Jump Method', + description=( + "The method to call to jump to the relation. Required."), + default="Base_jumpToRelatedDocument", + required=1) + + allow_jump = fields.CheckBoxField('allow_jump', + title='Allow Jump', + description=( + "Do we allow to jump to the relation ?"), + default=1, + required=0) + + base_category = fields.StringField('base_category', + title='Base Category', + description=( + "The method to call to set the relation. Required."), + default="", + required=1) + + portal_type = fields.ListTextAreaField('portal_type', + title='Portal Type', + description=( + "The method to call to set the relation. Required."), + default="", + required=1) + + allow_creation = fields.CheckBoxField('allow_creation', + title='Allow Creation', + description=( + "Do we allow to create new objects ?"), + default=1, + required=0) + + container_getter_id = fields.StringField('container_getter_id', + title='Container Getter Method', + description=( + "The method to call to get a container object."), + default="", + required=0) + + catalog_index = fields.StringField('catalog_index', + title='Catalog Index', + description=( + "The method to call to set the relation. Required."), + default="", + required=1) + + # XXX Is it a good idea to keep such a field ?? + # User can redefine setter method with a script (and so, don't use the API) + relation_setter_id = fields.StringField('relation_setter_id', + title='Relation Update Method', + description=( + "The method to invoke in order to update the relation"), + default="", + required=0) + + size = fields.IntegerField('size', + title='Size', + description=( + "The display size in rows of the field. If set to 1, the " + "widget will be displayed as a drop down box by many browsers, " + "if set to something higher, a list will be shown. Required."), + default=1, + required=1) + + columns = fields.ListTextAreaField('columns', + title="Columns", + description=( + "A list of attributes names to display."), + default=[], + required=0) + + sort = fields.ListTextAreaField('sort', + title='Default Sort', + description=('The default sort keys and order'), + default=[], + required=0) + + parameter_list = fields.ListTextAreaField('parameter_list', + title="Parameter List", + description=( + "A list of paramters used for the portal_catalog."), + default=[], + required=0) + + list_method = fields.MethodField('list_method', + title='List Method', + description=('The method to use to list' + 'objects'), + default='', + required=0) + + # delete double in order to keep a usable ZMI... + # XXX need to keep order ! + #property_names = dict([(i,0) for i in property_names]).keys() + _v_dict = {} + _v_property_name_list = [] + for property_name in property_names: + if not _v_dict.has_key(property_name): + _v_property_name_list.append(property_name) + _v_dict[property_name] = 1 + property_names = _v_property_name_list + + default_widget_rendering_instance = Widget.LinesTextAreaWidgetInstance + + def _generateRenderValueList(self, field, key, value_list, REQUEST): + result_list = [] + need_validation = 0 + #################################### + # Check value + #################################### + if isinstance(value_list, StringType): + # Value is a string, reformat it correctly + value_list = value_list.split("\n") + # Check all relation + for i in range(len(value_list)): + ################################### + # Sub field + ################################### + relation_field_id = field.generate_subfield_key("%s_%s" % \ + (SUB_FIELD_ID, i), + key=key) + relation_item_id = field.generate_subfield_key("%s_%s" % \ + (ITEM_ID, i), + key=key) + relation_item_list = REQUEST.get(relation_item_id, None) + value = value_list[i] + if (relation_item_list is not None) and \ + (value != ''): + need_validation = 1 + if value is None : + # rather than displaying nothing, display a marker when the + # property is not set + # XXX Translate ? + value = '??? (no value)' + # If we get a empty string, display nothing ! + if value != '': + result_list.append((Widget.TextWidgetInstance, relation_field_id, + relation_item_list, value, i)) + if not need_validation: + ################################### + # Main field + ################################### + result_list = [(Widget.LinesTextAreaWidgetInstance, None, [], + value_list, None)] + return result_list + + def render(self, field, key, value, REQUEST): """ - RelationStringField widget - - Works like a string field but includes one buttons - - - one search button which updates the field and sets a relation - - - creates object if not there - + Render text input field. """ - property_names = Widget.LinesTextAreaWidget.property_names + \ - RelationField.RelationStringFieldWidget.property_names - - # delete double in order to keep a usable ZMI... - #property_names = dict([(i,0) for i in property_names]).keys() # XXX need to keep order ! - _v_dict = {} - _v_property_name_list = [] - for property_name in property_names: - if not _v_dict.has_key(property_name): - _v_property_name_list.append(property_name) - _v_dict[property_name] = 1 - property_names = _v_property_name_list - - - def render(self, field, key, value, REQUEST): - """ - Render text input field. - """ - here = REQUEST['here'] - - relation_field_id = 'relation_%s' % key - relation_item_id = 'item_%s' % key - - portal_url = getToolByName(here, 'portal_url') - portal_url_string = portal_url() - portal_object = portal_url.getPortalObject() - - if type(value) == type(''): - # Value is a string, reformat it correctly - value_list = string.split(value, "\r\n") - else: - value_list = value - - need_validation = 0 - # Check all relation - for i in range( len(value_list) ): - relation_field_id = 'relation_%s_%s' % ( key, i ) - relation_item_id = 'item_%s_%s' % ( key, i ) - if REQUEST.has_key(relation_item_id) and value_list[i] != '': - need_validation = 1 - break - - html_string = '' - if need_validation: - # Check all relation - for i in range( len(value_list) ): - value = value_list[i] - relation_field_id = 'relation_%s_%s' % ( key, i ) - relation_item_id = 'item_%s_%s' % ( key, i ) - - - # If we get a empty string, display nothing ! - if value == '': - pass - - else: - - html_string += Widget.TextWidget.render(self, field, key, value, REQUEST) - - if REQUEST.has_key(relation_item_id): - relation_item_list = REQUEST.get(relation_item_id) - - if relation_item_list != []: - # Define default tales on the fly - tales_expr = field.tales.get('items', None) - defined_tales = 0 - if not tales_expr: - defined_tales = 1 - from Products.Formulator.TALESField import TALESMethod - field.tales['items'] = TALESMethod('REQUEST/relation_item_list') - - - REQUEST['relation_item_list'] = relation_item_list - html_string += ' %s ' % Widget.ListWidget.render(self, - field, relation_field_id, None, REQUEST) - REQUEST['relation_item_list'] = None - - if defined_tales: - # Delete default tales on the fly - field.tales['items'] = None - - else: - html_string += ' <input type="image" src="%s/images/exec16.png" value="update..." name="%s/portal_selections/viewSearchRelatedDocumentDialog%s_%s:method"/>' \ - % (portal_url_string, portal_object.getPath(), field.aq_parent._v_relation_field_index, i) - - html_string += '<br/>' - + html_string = '' + relation_field_index = REQUEST.get('_v_relation_field_index') + render_parameter_list = self._generateRenderValueList( + field, key, value, + REQUEST) + #################################### + # Render subfield + #################################### + html_string_list = [] + for widget_instance, relation_field_id, relation_item_list, \ + value_instance, sub_index in render_parameter_list: + sub_html_string = widget_instance.render(field, key, + value_instance, REQUEST) + if relation_item_list is not None: + if relation_item_list != []: + #################################### + # Render listfield + #################################### + tales_expr = field.tales.get('items', None) + defined_tales = 0 + if not tales_expr: + defined_tales = 1 + from Products.Formulator.TALESField import TALESMethod + # XXX XXX Do not write in the ZODB + field.tales['items'] = TALESMethod('REQUEST/relation_item_list') + + REQUEST['relation_item_list'] = relation_item_list + sub_html_string += ' %s ' % \ + Widget.ListWidgetInstance.render( + field, relation_field_id, None, REQUEST) + REQUEST['relation_item_list'] = None + if defined_tales: + # Delete default tales on the fly + field.tales['items'] = None else: - clean_value_list = [] - for v in value_list : - # rather than displaying nothing, display a marker when the - # property is not set - if v is None : v = '??? (no value)' - clean_value_list += [v] - # no modification made, we can display only a lines text area widget - html_string += Widget.LinesTextAreaWidget.render(self, field, key, clean_value_list, REQUEST) - - html_string += ' <input type="image" src="%s/images/exec16.png" value="update..." name="%s/portal_selections/viewSearchRelatedDocumentDialog%s:method"/>' \ - % (portal_url_string, portal_object.getPath(), field.aq_parent._v_relation_field_index) - - if value_list not in ((), [], None, ['']) and value_list == field.get_value('default') and field.get_value('allow_jump') == 1 : - if REQUEST.get('selection_name') is not None: - html_string += ' <a href="%s?field_id=%s&form_id=%s&selection_name=%s&selection_index=%s"><img src="%s/images/jump.png"/></a>' \ - % (field.get_value('jump_method'), field.id, field.aq_parent.id, REQUEST.get('selection_name'), REQUEST.get('selection_index'),portal_url_string) - else: - html_string += ' <a href="%s?field_id=%s&form_id=%s"><img src="%s/images/jump.png"/></a>' \ - % (field.get_value('jump_method'), field.id, field.aq_parent.id,portal_url_string) - - relation_field_index = getattr(field.aq_parent, '_v_relation_field_index', 0) - field.aq_parent._v_relation_field_index = relation_field_index + 1 # Increase index - return html_string - - def render_view(self, field, value): - """ - Render text field. - """ - if field.get_value('allow_jump') == 0 : - return Widget.LinesTextAreaWidget.render_view(self, field, value) - - REQUEST = get_request() - here = REQUEST['here'] - - portal_url = getToolByName(here, 'portal_url') - portal_url_string = portal_url() - - # no modification made, we can display only a lines text area widget - html_string = Widget.LinesTextAreaWidget.render_view(self, field, value) - if value not in ((), [], None, ''): - if REQUEST.get('selection_name') is not None: - html_string += ' <a href="%s?field_id=%s&form_id=%s&selection_name=%s&selection_index=%s"><img src="%s/images/jump.png"/></a>' \ - % (field.get_value('jump_method'), field.id, field.aq_parent.id, REQUEST.get('selection_name'), REQUEST.get('selection_index'),portal_url_string) - else: - html_string += ' <a href="%s?field_id=%s&form_id=%s"><img src="%s/images/jump.png"/></a>' \ - % (field.get_value('jump_method'), field.id, field.aq_parent.id,portal_url_string) - - return html_string + #################################### + # Render wheel + #################################### + sub_html_string += self.render_wheel( + field, value_instance, REQUEST, + relation_index=relation_field_index, + sub_index=sub_index) + html_string_list.append(sub_html_string) + #################################### + # Generate html + #################################### + html_string = '<br/>'.join(html_string_list) + #################################### + # Render jump + #################################### + if (value == field.get_value('default')): + # XXX Default rendering with value... + relation_html_string = self.render_relation_link(field, value, + REQUEST) + if relation_html_string != '': + html_string += ' %s' % relation_html_string + #################################### + # Update relation field index + #################################### + REQUEST.set('_v_relation_field_index', relation_field_index + 1) + return html_string + + def render_view(self, field, value): + """ + Render read only field. + """ + html_string = self.default_widget_rendering_instance.render_view( + field, value) + REQUEST = get_request() + relation_html_string = self.render_relation_link(field, value, REQUEST) + if relation_html_string != '': + html_string += ' %s' % relation_html_string + return html_string + + def render_wheel(self, field, value, REQUEST, relation_index=0, + sub_index=None): + """ + Render wheel used to display a listbox + """ + here = REQUEST['here'] + portal_url = getToolByName(here, 'portal_url') + portal_url_string = portal_url() + portal_object = portal_url.getPortalObject() + if sub_index is None: + sub_index_string = '' + else: + sub_index_string = '_%s' % sub_index + return ' <input type="image" ' \ + 'src="%s/images/exec16.png" value="update..." ' \ + 'name="%s/portal_selections/viewSearchRelatedDocumentDialog%s%s' \ + ':method"/>' % \ + (portal_url_string, portal_object.getPath(), + relation_index, sub_index_string) + + def render_relation_link(self, field, value, REQUEST): + """ + Render link to the related object. + """ + html_string = '' + here = REQUEST['here'] + portal_url = getToolByName(here, 'portal_url') + portal_url_string = portal_url() + portal_object = portal_url.getPortalObject() + if (value not in ((), [], None, '')) and \ + (field.get_value('allow_jump') == 1): + # Keep the selection name in the URL + if REQUEST.get('selection_name') is not None: + selection_name_html = '&selection_name=%s&selection_index=%s' % \ + (REQUEST.get('selection_name'), REQUEST.get('selection_index')) + else: + selection_name_html = '' + # Generate plan link + html_string += '<a href="%s/%s?field_id=%s&form_id=%s%s">' \ + '<img src="%s/images/jump.png" />' \ + '</a>' % \ + (here.absolute_url(), + field.get_value('jump_method'), + field.id, field.aq_parent.id, + selection_name_html, + portal_url_string) + return html_string class MultiRelationEditor: """ A class holding all values required to update a relation """ - def __init__(self, field_id, base_category, portal_type, portal_type_item, key, relation_setter_id, relation_editor_list): - - + def __init__(self, field_id, base_category, + portal_type_list, + portal_type_item, key, relation_setter_id, + relation_editor_list): self.field_id = field_id self.base_category = base_category - self.portal_type = portal_type + self.portal_type_list = portal_type_list self.portal_type_item = portal_type_item self.key = key self.relation_setter_id = relation_setter_id self.relation_editor_list = relation_editor_list - def __call__(self, REQUEST): if self.relation_editor_list != None: value_list = [] - - for i, value, uid, display_text in self.relation_editor_list: + for value, uid, display_text, relation_key, item_key in \ + self.relation_editor_list: value_list.append(value) if uid is not None: # Decorate the request so that we can display # the select item in a popup - #relation_field_id = 'relation_%s_%s' % ( self.key, i ) - #relation_item_id = 'item_%s_%s' % ( self.key, i ) - relation_field_id = 'relation_field_%s_%s' % ( self.field_id, i ) - relation_item_id = 'item_field_%s_%s' % ( self.field_id, i ) - - + # XXX To be unified + relation_field_id = relation_key + relation_item_id = item_key REQUEST.set(relation_item_id, ((display_text, uid),)) + # XXX Is it useful ? REQUEST.set(relation_field_id, uid) - REQUEST.set(self.field_id, value_list) # XXX Dirty else: # Make sure no default value appears - #REQUEST.set(self.field_id[len('field_'):], None) REQUEST.set(self.field_id, None) # XXX Dirty def view(self): @@ -255,276 +412,367 @@ class MultiRelationEditor: relation_uid_list = [] relation_object_list = [] - - for i, value, uid, display_text in self.relation_editor_list: + for value, uid, display_text, relation_key, item_key in \ + self.relation_editor_list: if uid is not None: - if type(uid) is type('a') and uid.startswith(new_content_prefix): + if isinstance(uid, StringType) and \ + uid.startswith(NEW_CONTENT_PREFIX): # Create a new content - portal_type = uid[len(new_content_prefix):] + portal_type = uid[len(NEW_CONTENT_PREFIX):] portal_module = None for p_item in self.portal_type_item: if p_item[0] == portal_type: - portal_module = o.getPortalObject().getDefaultModuleId( p_item[0] ) + portal_module = o.getPortalObject().getDefaultModuleId( + p_item[0]) if portal_module is not None: - portal_module_object = getattr(o.getPortalObject(), portal_module) + portal_module_object = getattr(o.getPortalObject(), + portal_module) kw ={} - #kw[self.key] = value - kw[self.key] = string.join( string.split(value,'%'), '' ) + kw[self.key] = value.replace('%', '') kw['portal_type'] = portal_type kw['immediate_reindex'] = 1 new_object = portal_module_object.newContent(**kw) uid = new_object.getUid() else: raise - relation_uid_list.append(int(uid)) - - relation_object_list.append( o.portal_catalog.getObject(uid) ) - - #if relation_uid_list != []: + + relation_uid_list.append(int(uid)) + relation_object_list.append( o.portal_catalog.getObject(uid)) # Edit relation if self.relation_setter_id: relation_setter = getattr(o, self.relation_setter_id) - relation_setter((), portal_type=self.portal_type) - relation_setter( relation_uid_list , portal_type=self.portal_type) + relation_setter((), portal_type=self.portal_type_list) + relation_setter(relation_uid_list, + portal_type=self.portal_type_list) else: # we could call a generic method which create the setter method name - set_method_name = '_set'+convertToUpperCase(self.base_category)+'ValueList' - getattr(o, set_method_name)( relation_object_list , portal_type=self.portal_type) - - else: - # Nothing to do - pass -# # Delete relation -# if self.relation_setter_id: -# relation_setter = getattr(o, self.relation_setter_id) -# relation_setter((), portal_type=self.portal_type) -# else: -# o._setValueUids(self.base_category, (), portal_type=self.portal_type) + set_method_name = '_set%sValueList' % \ + convertToUpperCase(self.base_category) + getattr(o, set_method_name)(relation_object_list, + portal_type=self.portal_type_list) allow_class(MultiRelationEditor) +class MultiRelationStringFieldValidator(Validator.LinesValidator): + """ + Validation includes lookup of relared instances + """ + message_names = Validator.LinesValidator.message_names +\ + ['relation_result_too_long', 'relation_result_ambiguous', + 'relation_result_empty',] -class MultiRelationStringFieldValidator(Validator.LinesValidator, RelationField.RelationStringFieldValidator): - """ - Validation includes lookup of relared instances - """ - message_names = Validator.LinesValidator.message_names + \ - RelationField.RelationStringFieldValidator.message_names - - # delete double in order to keep a usable ZMI... - #message_names = dict([(i,0) for i in message_names]).keys() # XXX need to keep order ! - _v_dict = {} - _v_message_name_list = [] - for message_name in message_names: - if not _v_dict.has_key(message_name): - _v_message_name_list.append(message_name) - _v_dict[message_name] = 1 - message_names = _v_message_name_list - - def validate(self, field, key, REQUEST): - portal_type = map(lambda x:x[0],field.get_value('portal_type')) - portal_type_item = field.get_value('portal_type') - base_category = field.get_value( 'base_category') - - # If the value is different, build a query - portal_selections = getToolByName(field, 'portal_selections') - portal_catalog = getToolByName(field, 'portal_catalog') - - # Get the current value - value_list = Validator.LinesValidator.validate(self, field, key, REQUEST) + # XXX Do we need to translate here ? + relation_result_too_long = "Too many documents were found." + relation_result_ambiguous = "Select appropriate document in the list." + relation_result_empty = "No such document was found." -# if type(value_list) == type(''): -# value_list = [value_list] + # Relation field variable + editor = MultiRelationEditor + default_validator_instance = Validator.LinesValidatorInstance + def _generateItemUidList(self, field, key, relation_uid_list, REQUEST=None): + """ + Generate tuple... + """ + result_list = [] + for i in range(len(relation_uid_list)): + # Generate a Item id for each value. + relation_item_id = field.generate_subfield_key("%s_%s" % \ + (ITEM_ID, i), + key=key) + relation_uid = relation_uid_list[i] + result_list.append((relation_item_id, relation_uid, None)) + return result_list + + def _generateFieldValueList(self, field, key, + value_list, current_value_list): + """ + Generate list of value, item_key + """ + item_value_list = [] + if isinstance(current_value_list, StringType): + current_value_list = [current_value_list] + # Check value list + if not (checkSameKeys(value_list, current_value_list)): + for i in range(len(value_list)): + value = value_list[i] + relation_field_id = field.generate_subfield_key("%s_%s" % \ + (SUB_FIELD_ID, i), + key=key) + relation_item_id = field.generate_subfield_key("%s_%s" % \ + (ITEM_ID, i), + key=key) + item_value_list.append((relation_field_id, value, relation_item_id)) + # Make possible to delete the content of the field. + if item_value_list == []: + relation_field_id = field.generate_subfield_key("%s" % \ + SUB_FIELD_ID, key=key) + relation_item_key = field.generate_subfield_key(ITEM_ID, key=key) + item_value_list.append((relation_field_id, '', relation_item_key)) + return item_value_list + + def validate(self, field, key, REQUEST): + """ + Validate the field. + """ + raising_error_needed = 0 + relation_editor_list = None + # Get some tool + catalog_index = field.get_value('catalog_index') + portal_type_list = [x[0] for x in field.get_value('portal_type')] + portal_catalog = getToolByName(field, 'portal_catalog') + #################################### + # Check list input + #################################### + relation_field_id = field.generate_subfield_key("%s" % \ + SUB_FIELD_ID, key=key) + relation_uid_list = REQUEST.get(relation_field_id, None) + #################################### + # User clicked on the wheel + #################################### + need_to_revalidate = 1 + if relation_uid_list is not None: + need_to_revalidate = 0 + relation_editor_list = [] + for relation_item_id, relation_uid, value in \ + self._generateItemUidList(field, key, relation_uid_list, + REQUEST=REQUEST): + found = 0 + try: + related_object = portal_catalog.getObject(relation_uid) + display_text = str(related_object.getProperty(catalog_index)) + found = 1 + except ValueError: + # Catch the error raised when the uid is a string + if relation_uid.startswith(NEW_CONTENT_PREFIX): + ############################## + # New content was selected, but the + # form is not validated + ############################## + portal_type = relation_uid[len(NEW_CONTENT_PREFIX):] + translated_portal_type = Message(domain='erp5_ui', + message=portal_type) + message = Message( + domain='erp5_ui', message='New ${portal_type}', + mapping={'portal_type': translated_portal_type}) + display_text = message + else: + display_text = 'Object has been deleted' + ################################ + # Modify if user modified his value + ################################ + if (found == 1) and \ + (value != display_text): + relation_editor_list = None +# import pdb; pdb.set_trace() + need_to_revalidate = 1 + REQUEST.set(relation_field_id, None) + break + if value is None: + value = display_text + # Storing display_text as value is needed in this case + relation_editor_list.append((value, + relation_uid, display_text, + None, relation_item_id)) +# str(relation_uid), display_text, + #################################### + # User validate the form + #################################### + if need_to_revalidate == 1: +# else: + #################################### + # Check the default field + #################################### + value_list = self.default_validator_instance.validate(field, + key, REQUEST) # If the value is the same as the current field value, do nothing current_value_list = field.get_value('default') - if type(current_value_list) == type(''): - current_value_list = [current_value_list] - - catalog_index = field.get_value('catalog_index') - relation_setter_id = field.get_value('relation_setter_id') - - relation_field_id = 'relation_%s' % ( key ) - # we must know if user validate the form or click on the wheel button - relation_uid_list = REQUEST.get(relation_field_id, None) - relation_field_sub_id = 'relation_%s_0' % ( key ) - if checkSameKeys( value_list, current_value_list ) and (relation_uid_list is None) and (not REQUEST.has_key( relation_field_sub_id )): - # Will be interpreted by Editor as "do nothing" - return MultiRelationEditor(field.id, base_category, - portal_type, portal_type_item, - catalog_index, relation_setter_id, None) - else: - - relation_field_id = 'relation_%s' % ( key ) - - # We must be able to erase the relation - if (value_list == ['']) and (not REQUEST.has_key( relation_field_id )): - display_text = 'Delete the relation' - return MultiRelationEditor(field.id, base_category, portal_type, portal_type_item, catalog_index, relation_setter_id, []) -# return RelationEditor(key, base_category, portal_type, None, -# portal_type_item, catalog_index, value, relation_setter_id, display_text) - # Will be interpreted by Base_edit as "delete relation" (with no uid and value = '') - - if REQUEST.has_key( relation_field_id ): - # we must know if user validate the form or click on the wheel button - relation_uid_list = REQUEST.get(relation_field_id, None) - if relation_uid_list != None: - relation_editor_list = [] - for i in range( len(relation_uid_list) ): - - relation_item_id = 'item_%s_%s' % ( key, i ) - relation_uid = relation_uid_list[i] - - related_object = portal_catalog.getObject(relation_uid) + field_value_list = self._generateFieldValueList(field, key, value_list, + current_value_list) + if len(field_value_list) != 0: + #################################### + # Values were changed + #################################### + relation_editor_list = [] + for relation_field_id, value, relation_item_id in field_value_list: + if value == '': + #################################### + # User want to delete this line + #################################### + # Clean request if necessary + if REQUEST.has_key(relation_field_id): + for subdict_name in ['form', 'other']: + subdict = getattr(REQUEST, subdict_name) + if subdict.has_key(relation_field_id): + subdict.pop(relation_field_id) + display_text = 'Delete the relation' + relation_editor_list.append((value, None, + display_text, None, None)) + # XXX RelationField implementation +# # We must be able to erase the relation +# display_text = 'Delete the relation' +# # Will be interpreted by Base_edit as "delete relation" +# # (with no uid and value = '') +# relation_editor_list = [(value, None, +# display_text, None, None)] + else: + relation_uid = REQUEST.get(relation_field_id, None) +# need_to_revalidate = 1 + if relation_uid not in (None, ''): +# need_to_revalidate = 0 +# found = 0 + #################################### + # User selected in a popup menu + #################################### + if isinstance(relation_uid, (list, tuple)): + relation_uid = relation_uid[0] + try: + related_object = portal_catalog.getObject(relation_uid) + except ValueError: + # Catch the exception raised when the uid is a string + related_object = None if related_object is not None: display_text = str(related_object.getProperty(catalog_index)) +# found = 1 else: - display_text = 'Object has been deleted' - # Check - REQUEST.set(relation_item_id, ( (display_text, relation_uid), )) - # Storing display_text as value is needded in this case - relation_editor_list.append( (i, display_text, str(relation_uid), display_text) ) - - return MultiRelationEditor(field.id, base_category, portal_type, portal_type_item, catalog_index, relation_setter_id, relation_editor_list) - - - else: - # User validate the form - - relation_editor_list = [] - raising_error_needed = 0 - raising_error_value = '' - - # Check all relation - for i in range( len(value_list) ): - relation_field_id = 'relation_%s_%s' % ( key, i ) - relation_item_id = 'item_%s_%s' % ( key, i ) - - relation_uid = REQUEST.get(relation_field_id, None) - - value = value_list[i] - - - # If we get a empty string, delete this line - if value == '': - # Clean request if necessary - if REQUEST.has_key( relation_field_id): - for subdict_name in ['form', 'other']: - subdict = getattr(REQUEST, subdict_name) - if subdict.has_key(relation_field_id): - subdict.pop(relation_field_id) + ############################## + # New content was selected, but the + # form is not validated + ############################## + if relation_uid.startswith(NEW_CONTENT_PREFIX): + ############################## + # New content was selected, but the + # form is not validated + ############################## + portal_type = relation_uid[len(NEW_CONTENT_PREFIX):] + translated_portal_type = Message(domain='erp5_ui', + message=portal_type) + message = Message( + domain='erp5_ui', message='New ${portal_type}', + mapping={'portal_type': translated_portal_type}) + display_text = message + else: + display_text = 'Object has been deleted' +# ################################ +# # Modify if user modified his value +# ################################ +# if (found == 1) and \ +# (value != display_text): +# REQUEST.set(relation_field_id, None) +# need_to_revalidate = 1 +# else: +# # Check +# REQUEST.set(relation_item_id, ((display_text, relation_uid),)) +# relation_editor_list.append((value, str(relation_uid), +# display_text, relation_field_id, +# relation_item_id)) + REQUEST.set(relation_item_id, ((display_text, relation_uid),)) + relation_editor_list.append((value, str(relation_uid), + display_text, relation_field_id, + relation_item_id)) +# if need_to_revalidate == 1: else: - # Got a true value - - if relation_uid not in (None, ''): - # A value has been defined by the user in popup menu - if type(relation_uid) in (type([]), type(())): relation_uid = relation_uid[0] - try: - related_object = portal_catalog.getObject(relation_uid) - except ValueError: - # Catch the exception raised when the uid is a string - related_object = None + #################################### + # User validate the form for this line + #################################### + kw ={} + kw[catalog_index] = value + kw['portal_type'] = portal_type_list + kw['sort_on'] = catalog_index + # Get the query results + relation_list = portal_catalog(**kw) + relation_uid_list = [x.uid for x in relation_list] + menu_item_list = [] + if len(relation_list) >= MAX_SELECT: + # If the length is long, raise an error + # This parameter means we need listbox help + # XXX XXX XXX Do we need to delete it ? + REQUEST.set(relation_item_id, []) + raising_error_needed = 1 + raising_error_value = 'relation_result_too_long' + elif len(relation_list) == 1: + # If the length is 1, return uid + relation_uid = relation_uid_list[0] + related_object = portal_catalog.getObject(relation_uid) if related_object is not None: display_text = str(related_object.getProperty(catalog_index)) + # Modify the value, in order to let the user + # modify it later... + value = display_text else: display_text = 'Object has been deleted' - # Check - REQUEST.set(relation_item_id, ( (display_text, relation_uid), )) - relation_editor_list.append( (i, value, str(relation_uid), display_text) ) - + # XXX XXX XXX + REQUEST.set(relation_item_id, ((display_text, + relation_uid),)) + relation_editor_list.append((value, relation_uid, + display_text, None, + relation_item_id)) +# relation_editor_list.append((0, value, relation_uid, +# display_text, None, None)) + elif len(relation_list) == 0: + # If the length is 0, raise an error + if field.get_value('allow_creation') == 1 : + # XXX + for portal_type in portal_type_list: + translated_portal_type = Message(domain='erp5_ui', + message=portal_type) + message = Message( + domain='erp5_ui', message='New ${portal_type}', + mapping={'portal_type': translated_portal_type}) + menu_item_list.append((message, + '%s%s' % (NEW_CONTENT_PREFIX, + portal_type))) + REQUEST.set(relation_item_id, menu_item_list) + raising_error_needed = 1 + raising_error_value = 'relation_result_empty' else: - - kw ={} - kw[catalog_index] = value - kw['portal_type'] = portal_type - kw['sort_on'] = catalog_index - # Get the query results - relation_list = portal_catalog(**kw) - relation_uid_list = map(lambda x: x.uid, relation_list) - localizer = getToolByName( field - , 'Localizer' - , None - ) - # Prepare a menu - if localizer is not None: - N_ = localizer.erp5_ui.gettext - else : - N_ = lambda msg, **kw: msg - menu_item_list = [('', '')] - new_object_menu_item_list = [] - for p in portal_type: - new_object_menu_item_list += [ ( N_('New %s') % (N_(p)) - , '%s%s' % (new_content_prefix,p) - ) - ] - - if len(relation_list) >= MAX_SELECT: - # If the length is long, raise an error - # This parameter means we need listbox help - REQUEST.set(relation_item_id, []) - raising_error_needed = 1 - raising_error_value = 'relation_result_too_long' - - elif len(relation_list) == 1: - # If the length is 1, return uid - relation_uid = relation_uid_list[0] - related_object = portal_catalog.getObject(relation_uid) - if related_object is not None: - display_text = str(related_object.getProperty(catalog_index)) - else: - display_text = 'Object has been deleted' - - REQUEST.set(relation_item_id, ( (display_text, relation_uid), )) - relation_editor_list.append( (0, value, relation_uid, display_text) ) - - elif len(relation_list) == 0: - # If the length is 0, raise an error - if field.get_value('allow_creation') == 1 : - menu_item_list += new_object_menu_item_list - REQUEST.set(relation_item_id, menu_item_list) - raising_error_needed = 1 - raising_error_value = 'relation_result_empty' - - else: - # If the length is short, raise an error - # len(relation_list) < MAX_SELECT: - - #menu_item_list += [('-', '')] - menu_item_list += map(lambda x: (x.getObject().getProperty(catalog_index), x.uid), - relation_list) - REQUEST.set(relation_item_id, menu_item_list) - raising_error_needed = 1 - raising_error_value = 'relation_result_ambiguous' - - # validate MultiRelation field - if raising_error_needed: - # Raise error - self.raise_error(raising_error_value, field) - return value_list - else: - # Can return editor - return MultiRelationEditor(field.id, base_category, portal_type, portal_type_item, catalog_index, relation_setter_id, relation_editor_list) + # If the length is short, raise an error + # len(relation_list) < MAX_SELECT: + menu_item_list.extend([( + x.getObject().getProperty(catalog_index), + x.uid) for x in relation_list]) + REQUEST.set(relation_item_id, menu_item_list) + raising_error_needed = 1 + raising_error_value = 'relation_result_ambiguous' + + ##################################### + # Validate MultiRelation field + ##################################### + if raising_error_needed: + # Raise error + self.raise_error(raising_error_value, field) + return value_list + else: + # Can return editor + base_category = field.get_value('base_category') + portal_type_item = field.get_value('portal_type') + relation_setter_id = field.get_value('relation_setter_id') + return self.editor(field.id, + base_category, + portal_type_list, + portal_type_item, catalog_index, + relation_setter_id, relation_editor_list) MultiRelationStringFieldWidgetInstance = MultiRelationStringFieldWidget() MultiRelationStringFieldValidatorInstance = MultiRelationStringFieldValidator() class MultiRelationStringField(ZMIField): - meta_type = "MultiRelationStringField" - security = ClassSecurityInfo() - - widget = MultiRelationStringFieldWidgetInstance - validator = MultiRelationStringFieldValidatorInstance + meta_type = "MultiRelationStringField" + security = ClassSecurityInfo() - security.declareProtected('Access contents information', 'get_value') - def get_value(self, id, **kw): - """Get value for id. + widget = MultiRelationStringFieldWidgetInstance + validator = MultiRelationStringFieldValidatorInstance - Optionally pass keyword arguments that get passed to TALES - expression. - """ - if id in ('is_relation_field', 'is_multi_relation_field'): - result = 1 - else: - result = ZMIField.get_value(self, id, **kw) - return result + security.declareProtected('Access contents information', 'get_value') + def get_value(self, id, **kw): + """ + Get value for id. + Optionally pass keyword arguments that get passed to TALES + expression. + """ + if id in ('is_relation_field', 'is_multi_relation_field'): + result = 1 + else: + result = ZMIField.get_value(self, id, **kw) + return result diff --git a/product/ERP5Form/RelationField.py b/product/ERP5Form/RelationField.py index 05c4275722..86ce53074f 100755 --- a/product/ERP5Form/RelationField.py +++ b/product/ERP5Form/RelationField.py @@ -1,7 +1,8 @@ ############################################################################## # -# Copyright (c) 2002 Nexedi SARL and Contributors. All Rights Reserved. +# Copyright (c) 2002, 2006 Nexedi SARL and Contributors. All Rights Reserved. # Jean-Paul Smets-Solanes <jp@nexedi.com> +# Romain Courteaud <romain@nexedi.com> # # WARNING: This program as such is intended to be used by professional # programmers who take the whole responsability of assessing all potential @@ -31,448 +32,120 @@ from Products.Formulator.Field import ZMIField from Products.Formulator.DummyField import fields from Products.ERP5Type.Utils import convertToUpperCase from Products.CMFCore.utils import getToolByName -from Globals import get_request from Products.PythonScripts.Utility import allow_class from Products.ERP5Type.Message import Message - -import string - +from Products.ERP5Form import MultiRelationField +from Products.ERP5Form.MultiRelationField import MAX_SELECT, \ + NEW_CONTENT_PREFIX, \ + SUB_FIELD_ID, ITEM_ID +from types import StringType from AccessControl import ClassSecurityInfo from zLOG import LOG -MAX_SELECT = 30 # Max. number of catalog result -new_content_prefix = '_newContent_' - - -class RelationStringFieldWidget(Widget.TextWidget, Widget.ListWidget): - """ - RelationStringField widget - Works like a string field but includes one buttons - - one search button which updates the field and sets a relation - - creates object if not there - """ - property_names = Widget.TextWidget.property_names + \ - ['update_method', 'jump_method', 'allow_jump', 'base_category', - 'portal_type', 'allow_creation', 'container_getter_id', 'catalog_index', - 'relation_setter_id', 'columns','sort','parameter_list','list_method', - 'first_item', 'items', 'size', 'extra_item'] - - # XXX Field to remove... - update_method = fields.StringField('update_method', - title='Update Method', - description=( - "The method to call to set the relation. Required."), - default="Base_validateRelation", - required=1) - - jump_method = fields.StringField('jump_method', - title='Jump Method', - description=( - "The method to call to jump to the relation. Required."), - default="Base_jumpToRelatedDocument", - required=1) - - allow_jump = fields.CheckBoxField('allow_jump', - title='Allow Jump', - description=( - "Do we allow to jump to the relation ?"), - default=1, - required=0) - - base_category = fields.StringField('base_category', - title='Base Category', - description=( - "The method to call to set the relation. Required."), - default="", - required=1) - - portal_type = fields.ListTextAreaField('portal_type', - title='Portal Type', - description=( - "The method to call to set the relation. Required."), - default="", - required=1) - - allow_creation = fields.CheckBoxField('allow_creation', - title='Allow Creation', - description=( - "Do we allow to create new objects ?"), - default=1, - required=0) - - container_getter_id = fields.StringField('container_getter_id', - title='Container Getter Method', - description=( - "The method to call to get a container object."), - default="", - required=0) - - catalog_index = fields.StringField('catalog_index', - title='Catalog Index', - description=( - "The method to call to set the relation. Required."), - default="", - required=1) - - # XXX Is it a good idea to keep such a field ?? - # User can redefine setter method with a script (and so, don't use the API) - relation_setter_id = fields.StringField('relation_setter_id', - title='Relation Update Method', - description=( - "The method to invoke in order to update the relation"), - default="", - required=0) - - size = fields.IntegerField('size', - title='Size', - description=( - "The display size in rows of the field. If set to 1, the " - "widget will be displayed as a drop down box by many browsers, " - "if set to something higher, a list will be shown. Required."), - default=1, - required=1) - - columns = fields.ListTextAreaField('columns', - title="Columns", - description=( - "A list of attributes names to display."), - default=[], - required=0) - - sort = fields.ListTextAreaField('sort', - title='Default Sort', - description=('The default sort keys and order'), - default=[], - required=0) - - parameter_list = fields.ListTextAreaField('parameter_list', - title="Parameter List", - description=( - "A list of paramters used for the portal_catalog."), - default=[], - required=0) - - list_method = fields.MethodField('list_method', - title='List Method', - description=('The method to use to list' - 'objects'), - default='', - required=0) - - def render(self, field, key, value, REQUEST): - """Render text input field. - """ - relation_field_id = 'relation_%s' % key - relation_item_id = 'item_%s' % key - here = REQUEST['here'] - portal_url = getToolByName(here, 'portal_url') - portal_url_string = portal_url() - portal_object = portal_url.getPortalObject() - html_string = Widget.TextWidget.render(self, field, key, value, - REQUEST) - - if REQUEST.has_key(relation_item_id): - # Define default tales on the fly - tales_expr = field.tales.get('items', None) - if not tales_expr: - from Products.Formulator.TALESField import TALESMethod - field.tales['items'] = TALESMethod('REQUEST/relation_item_list') - REQUEST['relation_item_list'] = REQUEST.get(relation_item_id) - html_string += ' %s ' % Widget.ListWidget.render(self, - field, relation_field_id, None, REQUEST) - REQUEST['relation_item_list'] = None - - # We used to add a button which has a path reference to a base category... - # but it really created too many problems - # now we do it in another way - # we compare what has been changed in the relation update script - - #elif value != field.get_value('default'): - else: - html_string += \ - ' <input type="image" ' \ - 'src="%s/images/exec16.png" value="update..." ' \ - 'name="%s/portal_selections/viewSearchRelatedDocumentDialog%s' \ - ':method"/>' % \ - (portal_url_string, portal_object.getPath(), - getattr(field.aq_parent, '_v_relation_field_index', 0)) - relation_field_index = getattr(field.aq_parent, - '_v_relation_field_index', 0) - # Increase index - field.aq_parent._v_relation_field_index = relation_field_index + 1 - -# import pdb; pdb.set_trace() -# field.get_value('default') - if (value not in ( None, '' )) and \ - (not REQUEST.has_key(relation_item_id)) and \ - (value == field.get_value('default')) and \ - (field.get_value('allow_jump') == 1): - if REQUEST.get('selection_name') is not None: - html_string += ' <a href="%s/%s?field_id=%s&' \ - 'form_id=%s&selection_name=%s&selection_index=%s' \ - '"><img src="%s/images/jump.png"/></a>' % \ - (here.absolute_url(), field.get_value('jump_method'), - field.id, field.aq_parent.id, - REQUEST.get('selection_name'), - REQUEST.get('selection_index'), - portal_url_string) - else: - html_string += ' <a href="%s/%s?field_id=%s&' \ - 'form_id=%s"><img src="%s/images/jump.png"/></a>' \ - % (here.absolute_url(), field.get_value('jump_method'), - field.id, field.aq_parent.id,portal_url_string) - return html_string +class RelationStringFieldWidget( + MultiRelationField.MultiRelationStringFieldWidget): + """ + RelationStringField widget + Works like a string field but includes one buttons + - one search button which updates the field and sets a relation + - creates object if not there + """ + property_names = Widget.TextWidget.property_names + \ + MultiRelationField.MultiRelationStringFieldWidget.local_property_names + + default_widget_rendering_instance = Widget.TextWidgetInstance + + def _generateRenderValueList(self, field, key, value, REQUEST): + relation_field_id = field.generate_subfield_key(SUB_FIELD_ID, key=key) + relation_item_key = field.generate_subfield_key(ITEM_ID, key=key) + relation_item_list = REQUEST.get(relation_item_key, []) + return [(Widget.TextWidgetInstance, relation_field_id, + relation_item_list, value, None)] + +class RelationEditor(MultiRelationField.MultiRelationEditor): + """ + A class holding all values required to update a relation + """ + def __call__(self, REQUEST): + MultiRelationField.MultiRelationEditor.__call__(self, REQUEST) + value = REQUEST.get(self.field_id) + if value is not None: + REQUEST.set(self.field_id, value[0]) - def render_view(self, field, value): - """Render text input field. - """ - if field.get_value('allow_jump') == 0 : - return Widget.TextWidget.render_view(self, field, value) - REQUEST = get_request() - here = REQUEST['here'] - html_string = Widget.TextWidget.render_view(self, field, value) - portal_url_string = getToolByName(here, 'portal_url')() - if value not in ('', None): - html_string = '<a href="%s/%s?field_id=%s&form_id=%s">%s</a>' \ - % (here.absolute_url(), field.get_value('jump_method'), - field.id, field.aq_parent.id, html_string) - html_string += ' <a href="%s/%s?field_id=%s&form_id=%s">' \ - '<img src="%s/images/jump.png"/></a>' % \ - (here.absolute_url(), field.get_value('jump_method'), - field.id, field.aq_parent.id, portal_url_string) - return html_string +allow_class(RelationEditor) -class RelationEditor: +class RelationStringFieldValidator( + MultiRelationField.MultiRelationStringFieldValidator): + """ + Validation includes lookup of relared instances + """ + + message_names = Validator.StringValidator.message_names + \ + MultiRelationField.MultiRelationStringFieldValidator.message_names + # Delete double in order to keep a usable ZMI... + # Need to keep order ! + _v_dict = {} + _v_message_name_list = [] + for message_name in message_names: + if not _v_dict.has_key(message_name): + _v_message_name_list.append(message_name) + _v_dict[message_name] = 1 + message_names = _v_message_name_list + + # Relation field variable + editor = RelationEditor + default_validator_instance = Validator.StringValidatorInstance + + def _generateItemUidList(self, field, key, relation_uid_list, REQUEST=None): """ - A class holding all values required to update a relation + Generate list of uid, item_key """ - - def __init__(self, field_id, base_category, portal_type, uid, - portal_type_item, key, value, relation_setter_id, - container_getter_id, display_text): - self.field_id = field_id - self.uid = uid - self.base_category = base_category - self.portal_type = portal_type - self.portal_type_item = portal_type_item - self.key = key - self.value = value - self.relation_setter_id = relation_setter_id - self.container_getter_id = container_getter_id - self.display_text = display_text - - def __call__(self, REQUEST): - if self.uid is not None: - # Decorate the request so that we can display - # the select item in a popup - relation_field_id = 'relation_%s' % self.field_id - relation_item_id = 'item_%s' % self.field_id - REQUEST.set(relation_item_id, ((self.display_text, self.uid),)) - REQUEST.set(relation_field_id, self.uid) - # XXX Dirty - REQUEST.set(self.field_id[len('field_'):], self.value) - else: - # Make sure no default value appears - REQUEST.set(self.field_id[len('field_'):], None) - - def view(self): - return self.__dict__ - - def edit(self, o): - if self.uid is not None: - if type(self.uid) is type('a') and \ - self.uid.startswith(new_content_prefix): - # Create a new content - portal_type = self.uid[len(new_content_prefix):] - container = None - for p_item in self.portal_type_item: - if p_item[0] == portal_type: - if self.container_getter_id: - container = getattr(o, self.container_getter_id)( - portal_type=portal_type) - else: - portal_module = o.getPortalObject().getDefaultModuleId( p_item[0] ) - container = getattr(o.getPortalObject(), portal_module) - if container is not None: - kw ={} - kw[self.key] = string.join( string.split(self.value,'%'), '' ) - kw['portal_type'] = portal_type - kw['immediate_reindex'] = 1 - new_object = container.newContent(**kw) - self.uid = new_object.getUid() - else: - raise - - # Edit relation - if self.relation_setter_id: - relation_setter = getattr(o, self.relation_setter_id) - relation_setter((), portal_type=self.portal_type) - relation_setter((int(self.uid),), portal_type=self.portal_type) - else: - # we could call a generic method which create the setter method name - set_method_name = '_set'+convertToUpperCase(self.base_category)+'Value' - object = o.portal_catalog.getObject( self.uid ) - getattr(o, set_method_name)( object,portal_type=self.portal_type ) - - else: - if self.value == '': - # Delete relation - if self.relation_setter_id: - relation_setter = getattr(o, self.relation_setter_id) - relation_setter((), portal_type=self.portal_type) - else: - # we could call a generic method which create the setter method name - set_method_name = '_set'+convertToUpperCase(self.base_category) - getattr(o, set_method_name)( None ,portal_type=self.portal_type) - - -allow_class(RelationEditor) - -class RelationStringFieldValidator(Validator.StringValidator): + relation_item_id = field.generate_subfield_key(ITEM_ID, + key=key) + if isinstance(relation_uid_list, (list, tuple)): + try: + relation_uid_list = relation_uid_list[0] + except IndexError: + # No object was selected + return [] + value = self.default_validator_instance.validate(field, + key, REQUEST) + return [(relation_item_id, relation_uid_list, value)] + + def _generateFieldValueList(self, field, key, + value_list, current_value_list): """ - Validation includes lookup of relared instances + Generate list of value, item_key """ - - message_names = Validator.StringValidator.message_names +\ - ['relation_result_too_long', 'relation_result_ambiguous', - 'relation_result_empty',] - - relation_result_too_long = "Too many documents were found." - relation_result_ambiguous = "Select appropriate document in the list." - relation_result_empty = "No such document was found." - - def validate(self, field, key, REQUEST): - relation_field_id = 'relation_%s' % key - relation_item_id = 'item_%s' % key - portal_type = map(lambda x:x[0],field.get_value('portal_type')) - portal_type_item = field.get_value('portal_type') - base_category = field.get_value( 'base_category') - # If the value is different, build a query - portal_selections = getToolByName(field, 'portal_selections') - portal_catalog = getToolByName(field, 'portal_catalog') - # Get the current value - value = Validator.StringValidator.validate(self, field, key, REQUEST) - # If the value is the same as the current field value, do nothing - current_value = field.get_value('default') - # If a relation has been defined in a popup menu, use it - relation_uid = REQUEST.get(relation_field_id, None) - catalog_index = field.get_value('catalog_index') - parameter_list = field.get_value('parameter_list') - relation_setter_id = field.get_value('relation_setter_id') - container_getter_id = field.get_value('container_getter_id') - - if (value == current_value) and (relation_uid is None): - # Will be interpreted by Editor as "do nothing" - return RelationEditor(key, base_category, portal_type, None, - portal_type_item, catalog_index, None, - relation_setter_id, container_getter_id, None) - if relation_uid not in (None, ''): - # A value has been defined by the user - if type(relation_uid) in (type([]), type(())): - if len( relation_uid ) == 0: - # No object was selected... - return None - else: - relation_uid = relation_uid[0] - - try: - related_object = portal_catalog.getObject(relation_uid) - except ValueError: - # Catch the error raised when the uid is a string - related_object = None - if related_object is not None: - display_text = str(related_object.getProperty(catalog_index)) - else: - display_text = 'Object has been deleted' - return RelationEditor(key, base_category, portal_type, relation_uid, - portal_type_item, catalog_index, value, - relation_setter_id, container_getter_id, - display_text) - - # We must be able to erase the relation - if value == '': - display_text = 'Delete the relation' - # Will be interpreted by Base_edit as "delete relation" - # (with no uid and value = '') - return RelationEditor(key, base_category, portal_type, None, - portal_type_item, catalog_index, value, - relation_setter_id, container_getter_id, display_text) - - kw ={} - kw[catalog_index] = value - kw['portal_type'] = portal_type - kw['sort_on'] = catalog_index - if len(parameter_list) > 0: - for k,v in parameter_list: - kw[k] = v - # Get the query results - relation_list = portal_catalog(**kw) - relation_uid_list = map(lambda x: x.uid, relation_list) - # Prepare a menu - menu_item_list = [('', '')] - new_object_menu_item_list = [] - for p in portal_type: - translated_p = Message(domain='erp5_ui',message=p) - message = Message(domain='erp5_ui',message = 'New ${portal_type}', - mapping={'portal_type':translated_p}) - new_object_menu_item_list += [ ( message - , '%s%s' % (new_content_prefix,p) - ) - ] - # If the length is 1, return uid - if len(relation_list) == 1: - relation_uid = relation_uid_list[0] - related_object = portal_catalog.getObject(relation_uid) - if related_object is not None: - display_text = str(related_object.getProperty(catalog_index)) - else: - display_text = 'Object has been deleted' - - return RelationEditor( - key, base_category, portal_type, relation_uid, - portal_type_item, catalog_index, value, - relation_setter_id, container_getter_id, display_text) - # If the length is 0, raise an error - elif len(relation_list) == 0: - if field.get_value('allow_creation') == 1 : - menu_item_list += new_object_menu_item_list - REQUEST.set(relation_item_id, menu_item_list) - self.raise_error('relation_result_empty', field) - # If the length is short, raise an error - elif len(relation_list) < MAX_SELECT: - #menu_item_list += [('-', '')] - menu_item_list.extend([(x.getObject().getProperty(catalog_index), - x.uid) for x in relation_list]) - REQUEST.set(relation_item_id, menu_item_list) - self.raise_error('relation_result_ambiguous', field) - else: - # If the length is long, raise an error - - # If this error is raise, we don t want to create a new object... - #REQUEST.set(relation_item_id, menu_item_list) - self.raise_error('relation_result_too_long', field) + if value_list == current_value_list: + return [] + else: + relation_field_id = field.generate_subfield_key("%s" % \ + SUB_FIELD_ID, key=key) + relation_item_key = field.generate_subfield_key(ITEM_ID, key=key) + return [(relation_field_id, value_list, relation_item_key)] RelationStringFieldWidgetInstance = RelationStringFieldWidget() RelationStringFieldValidatorInstance = RelationStringFieldValidator() class RelationStringField(ZMIField): - meta_type = "RelationStringField" - security = ClassSecurityInfo() - - widget = RelationStringFieldWidgetInstance - validator = RelationStringFieldValidatorInstance + meta_type = "RelationStringField" + security = ClassSecurityInfo() - security.declareProtected('Access contents information', 'get_value') - def get_value(self, id, **kw): - """Get value for id. + widget = RelationStringFieldWidgetInstance + validator = RelationStringFieldValidatorInstance - Optionally pass keyword arguments that get passed to TALES - expression. - """ - if id == 'is_relation_field': - result = 1 - elif id == 'is_multi_relation_field': - result = 0 - else: - result = ZMIField.get_value(self, id, **kw) - return result + security.declareProtected('Access contents information', 'get_value') + def get_value(self, id, **kw): + """ + Get value for id. + Optionally pass keyword arguments that get passed to TALES + expression. + """ + if id == 'is_relation_field': + result = 1 + elif id == 'is_multi_relation_field': + result = 0 + else: + result = ZMIField.get_value(self, id, **kw) + return result diff --git a/product/ERP5Form/SelectionTool.py b/product/ERP5Form/SelectionTool.py index 9718b05e35..7b48f3f3b8 100755 --- a/product/ERP5Form/SelectionTool.py +++ b/product/ERP5Form/SelectionTool.py @@ -52,6 +52,7 @@ import random import string from zLOG import LOG from Acquisition import Implicit, aq_base +from Products.ERP5Type.Message import Message class SelectionError( Exception ): @@ -920,7 +921,7 @@ class SelectionTool( UniqueObject, SimpleItem ): return object # Related document searching - def viewSearchRelatedDocumentDialog(self, index, form_id, REQUEST=None, + def viewSearchRelatedDocumentDialog(self, index, form_id, REQUEST=None, sub_index=None, **kw): """ Returns a search related document dialog @@ -928,8 +929,7 @@ class SelectionTool( UniqueObject, SimpleItem ): """ if sub_index != None: REQUEST.form['sub_index'] = sub_index - - # Find the object which needs to be updated + # Find the object which needs to be updated object_uid = REQUEST.get('object_uid', None) object_path = REQUEST.get('object_path', None) if object_uid is not None: @@ -942,72 +942,56 @@ class SelectionTool( UniqueObject, SimpleItem ): if object_path is not None: o = self.getPortalObject().restrictedTraverse(object_path) if o is not None: + # XXX o.immediateReindexObject() object_uid = o.getUid() else: - return "Sorrry, Error, the calling object was not catalogued. " \ - "Do not know how to do ?" - + raise SelectionError, \ + "Sorrry, Error, the calling object was not catalogued. " \ + "Do not know how to do ?" # Find the field which was clicked on # Important to get from the object instead of self form = getattr(o, form_id) field = None + # Search the correct field + relation_field_found = 0 relation_index = 0 - - # find the correct field - field_list = [] # XXX may be should support another parameter, - # like include_non_editable=0 for field in form.get_fields(include_disabled=0): - if field.get_value('editable',REQUEST=REQUEST): - field_list.append(field) - - relation_field_found = 0 - for field in field_list: - try: - dumb = field.get_value('is_relation_field') - # XXX FIXME Exception name is not in locals. - # This can be related to a bad python file import - # I already had this kind of error with another python software, - # and the only solution I found was to use ihooks to - # import python files. - # I have to check this. -# except KeyError: - except: - pass -# relation_index += 1 - else: - if index == relation_index: - relation_field_found = 1 - break + if field.get_value('editable', REQUEST=REQUEST): + try: + dumb = field.get_value('is_relation_field') + except: + # except KeyError: + # XXX FIXME Exception name is not in locals. + # Namespace seems a bit broken... + LOG("SelectionTool", 0, "Exception catched with broken namespace!") + pass else: - relation_index += 1 - -# if getattr(field, 'is_relation_field', None): -# if index == relation_index: -# relation_field_found = 1 -# break -# else: -# relation_index += 1 - + if index == relation_index: + relation_field_found = 1 + break + else: + relation_index += 1 if not relation_field_found: + # We didn't find the field... raise SelectionError, "SelectionTool: can not find the relation" \ " field %s" % index - - field_value = REQUEST.form['field_%s' % field.id] - - selection_name = 'Base_viewRelatedObjectList' - - # reselt current selection - self.portal_selections.setSelectionFor(selection_name, None) - - # XXX portal_status_message = - # "Please select one object to precise the value: - # '%s' in the field: '%s'" % (field_value, field.get_orig_value('title')) - portal_status_message = "Please select one object." - - if field.get_value('is_multi_relation_field'): - if sub_index is None: + else: + # Field found + field_key = field.generate_field_key() + field_value = REQUEST.form[field_key] + # XXX Hardcoded form name + redirect_form_id = 'Base_viewRelatedObjectList' + redirect_form = getattr(o, redirect_form_id) + # XXX Hardcoded listbox field + selection_name = redirect_form.listbox.get_value('selection_name') + # Reset current selection + self.portal_selections.setSelectionFor(selection_name, None) + + + if (field.get_value('is_multi_relation_field')) and \ + (sub_index is None): # user click on the wheel, not on the validation button # we need to facilitate user search @@ -1029,110 +1013,60 @@ class SelectionTool( UniqueObject, SimpleItem ): self.portal_selections.setSelectionCheckedUidsFor( selection_name, current_uid_list) - - REQUEST.form['field_%s' % field.id] = field_value - # XXX portal_status_message = - # "Please select one or more object to define the field: - # '%s'" % field.get_orig_value('title') - portal_status_message = "Please select one (or more) object." + # XXX +# field_value = '' + field_value = str(field_value).splitlines() + REQUEST.form[field_key] = field_value + portal_status_message = Message( + domain='erp5_ui', + message="Please select one (or more) object.") + else: + portal_status_message = Message(domain='erp5_ui', + message="Please select one object.") - # Save the current REQUEST form - # We can't put FileUpload instances because we can't pickle them - pickle_kw = {} - for key in REQUEST.form.keys(): - if not isinstance(REQUEST.form[key],FileUpload): - pickle_kw[key] = REQUEST.form[key] + # Save the current REQUEST form + # We can't put FileUpload instances because we can't pickle them + pickle_kw = {} + for key in REQUEST.form.keys(): + if not isinstance(REQUEST.form[key],FileUpload): + pickle_kw[key] = REQUEST.form[key] - form_pickle, form_signature = self.getPickleAndSignature(**pickle_kw) - REQUEST.form_pickle = form_pickle - REQUEST.form_signature = form_signature - - base_category = None - kw = {} - - kw['object_uid'] = object_uid - kw['form_id'] = 'Base_viewRelatedObjectList' - kw['selection_name'] = 'Base_viewRelatedObjectList' - kw['selection_index'] = 0 # We start on the first page - kw['field_id'] = field.id - kw['portal_type'] = map(lambda x:x[0],field.get_value('portal_type')) - parameter_list = field.get_value('parameter_list') - if len(parameter_list) > 0: - for k,v in parameter_list: - kw[k] = v - kw['reset'] = 0 - kw['base_category'] = field.get_value( 'base_category') - kw['cancel_url'] = REQUEST.get('HTTP_REFERER') - kw['previous_form_id'] = form_id - - - kw[field.get_value('catalog_index')] = str(field_value).splitlines() - - """ - # We work with strings - ie. single values - kw ={} - context.portal_selections.setSelectionParamsFor('Base_viewRelatedObjectList', kw.copy()) - previous_uids = o.getValueUids(base_category, portal_type=portal_type) - relation_list = context.portal_catalog(**kw) - relation_uid_list = map(lambda x: x.uid, relation_list) - uids = [] - """ - - # Need to redirect, if we want listbox nextPage to work - kw['form_pickle'] = form_pickle - kw['form_signature'] = form_signature - kw['portal_status_message'] = portal_status_message - - redirect_url = '%s/%s?%s' % ( o.absolute_url() - , 'Base_viewRelatedObjectList' - , make_query(kw) - ) - - REQUEST[ 'RESPONSE' ].redirect( redirect_url ) - -# # Empty the selection (uid) -# REQUEST.form = kw # New request form -# -# # Define new HTTP_REFERER -# REQUEST.HTTP_REFERER = '%s/Base_viewRelatedObjectList' % o.absolute_url() -# -# # Return the search dialog -# return o.Base_viewRelatedObjectList(REQUEST=REQUEST) - - - # XXX do not use this method, use aq_dynamic (JPS) -# def __getattr__(self, name): -# dynamic_method_name = 'viewSearchRelatedDocumentDialog' -# if name[:len(dynamic_method_name)] == dynamic_method_name: -# method_count_string = name[len(dynamic_method_name):] -# # be sure that method name is correct -# try: -# import string -# method_count = string.atoi(method_count_string) -# except: -# raise AttributeError, name -# else: -# # generate dynamicaly needed forwarder methods -# def viewSearchRelatedDocumentDialogWrapper(self, form_id, REQUEST=None, **kw): -# """ -# viewSearchRelatedDocumentDialog Wrapper -# """ -# return self.viewSearchRelatedDocumentDialog(method_count, form_id, REQUEST=REQUEST, **kw) -# -# setattr(self.__class__, name, viewSearchRelatedDocumentDialogWrapper) -# -# klass = self.__class__ -# if hasattr(klass, 'security'): -# from Products.ERP5Type import Permissions as ERP5Permissions -# klass.security.declareProtected(ERP5Permissions.View, name) -# else: -# # XXX security declaration always failed.... -# LOG('WARNING ERP5Form SelectionTool, security not defined on',0,klass.__name__) -# -# return getattr(self, name) -# else: -# raise AttributeError, name + form_pickle, form_signature = self.getPickleAndSignature(**pickle_kw) + REQUEST.form_pickle = form_pickle + REQUEST.form_signature = form_signature + + base_category = None + kw = {} + kw['object_uid'] = object_uid + kw['form_id'] = redirect_form_id + kw['selection_name'] = selection_name + kw['selection_index'] = 0 # We start on the first page + kw['field_id'] = field.id + kw['portal_type'] = [x[0] for x in field.get_value('portal_type')] + parameter_list = field.get_value('parameter_list') + if len(parameter_list) > 0: + for k,v in parameter_list: + kw[k] = v + kw['reset'] = 0 + kw['base_category'] = field.get_value( 'base_category') + kw['cancel_url'] = REQUEST.get('HTTP_REFERER') + kw['previous_form_id'] = form_id + + # XXX +# kw[field.get_value('catalog_index')] = str(field_value).splitlines() + kw[field.get_value('catalog_index')] = field_value + # Need to redirect, if we want listbox nextPage to work + kw['form_pickle'] = form_pickle + kw['form_signature'] = form_signature + kw['portal_status_message'] = portal_status_message + + redirect_url = '%s/%s?%s' % ( o.absolute_url() + , redirect_form_id + , make_query(kw) + ) + + REQUEST[ 'RESPONSE' ].redirect( redirect_url ) def _aq_dynamic(self, name): """ @@ -1141,14 +1075,13 @@ class SelectionTool( UniqueObject, SimpleItem ): """ aq_base_name = getattr(aq_base(self), name, None) if aq_base_name == None: - dynamic_method_name = 'viewSearchRelatedDocumentDialog' + DYNAMIC_METHOD_NAME = 'viewSearchRelatedDocumentDialog' + method_name_length = len(DYNAMIC_METHOD_NAME) + zope_security = '__roles__' - if (name[:len(dynamic_method_name)] == dynamic_method_name) and \ + if (name[:method_name_length] == DYNAMIC_METHOD_NAME) and \ (name[-len(zope_security):] != zope_security): - - method_count_string_list = string.split( - name[len(dynamic_method_name):], - '_') + method_count_string_list = name[method_name_length:].split('_') method_count_string = method_count_string_list[0] # be sure that method name is correct try: @@ -1173,15 +1106,9 @@ class SelectionTool( UniqueObject, SimpleItem ): """ LOG('SelectionTool.viewSearchRelatedDocumentDialogWrapper, kw', 0, kw) - if sub_index == None: - return self.viewSearchRelatedDocumentDialog( - method_count, form_id, - REQUEST=REQUEST, **kw) - else: - return self.viewSearchRelatedDocumentDialog( + return self.viewSearchRelatedDocumentDialog( method_count, form_id, REQUEST=REQUEST, sub_index=sub_index, **kw) - setattr(self.__class__, name, viewSearchRelatedDocumentDialogWrapper) -- 2.30.9