# (C) Copyright 2004 Nexedi SARL <http://nexedi.com>
# Authors: Sebastien Robin <seb@nexedi.com>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License version 2 as published
# by the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
# 02111-1307, USA.
#

from Products.CPSDocument.CPSDocument import CPSDocument
from Products.CPSSchemas.BasicFields import CPSImageField, CPSFileField, CPSDateTimeField
from Products.CPSSchemas.BasicFields import CPSStringField, CPSIntField
from Products.ERP5Type.Base import Base
from Products.ERP5Type.Utils import UpperCase
from Acquisition import aq_base, aq_inner
from AccessControl import ClassSecurityInfo
from Products.CMFCore.CMFCorePermissions import View
from zLOG import LOG

class PatchedCPSDocument(CPSDocument):

  security = ClassSecurityInfo()

  security.declareProtected( View, '_propertyMap' )
  def _propertyMap(self):
    """
      Returns fake property sheet
    """
    property_sheet = []
    property_sheet.append(
      {
        'id'    :   'layout_and_schema',
        'type'  :   'object'
      }
      )
    type_info = self.getTypeInfo()
    field_list = []
    if type_info is not None:
      data_model = type_info.getDataModel(self)
      if data_model is not None:
        field_list = data_model._fields.items()
    field_list.sort()
    for (prop_id,field) in field_list:
      #for field in schema.objectValues():
      #LOG('testjp',0,'field: %s' % str(field))
      f_type = None
      #for p in field._properties:
      #  if p['id'] == 'default':
      #    f_type = p['type']
      if isinstance(field,CPSImageField):
        f_type = 'object'
      elif isinstance(field,CPSStringField):
        f_type = 'string'
      elif isinstance(field,CPSDateTimeField):
        f_type = 'date'
      elif isinstance(field,CPSFileField):
        f_type = 'object'
      elif isinstance(field,CPSIntField):
        f_type = 'int'
      elif isinstance(field,CPSDocument):
        pass
      if prop_id.find('attachedFile')==0:
        f_type='object'  # In a flexible content, we do have some attachedFile_f1
                         # which are CPStringFiels with some binary data
                         # XXX This should NOT BE NEEDED!!
      if f_type is not None:
        property_sheet.append(
          {
            'id'    :   prop_id,
            'type'  :   f_type
          }
          )
    return tuple(property_sheet + list(getattr(self, '_local_properties', ())))


  security.declareProtected( View, 'getProperty' )
  def getProperty(self, key, d=None):
    """
      Previous Name: getValue

      Generic accessor. Calls the real accessor
    """
    data_model = self.getTypeInfo().getDataModel(self)
    accessor_name = 'get' + UpperCase(key)
    base = aq_base(self)
    if data_model.has_key(key):
      return data_model.get(key)
    elif hasattr(base,accessor_name):
      method = getattr(base,accessor_name)
      return method()
    return None

  security.declarePrivate('getLayoutAndSchema' )
  def getLayoutAndSchema(self):
    if hasattr(self,'.cps_layouts') and hasattr(self,'.cps_schemas'):
      return (aq_base(self._getOb(".cps_layouts")),aq_base(self._getOb(".cps_schemas")))
    return None

  security.declarePrivate('setLayoutAndSchema' )
  def setLayoutAndSchema(self, data):
    """
    data must be : (layout,schema)
    """
    if data is not None:
      self._setOb(".cps_layouts",data[0])
      self._setOb(".cps_schemas",data[1])

  security.declarePrivate('_setProperty' )
  def _setProperty(self, key, value, type='string'):
    """
      Set the property for cps objects
    """
    LOG('PatchCPSDoc._setProperty',0,'key: %s, value: %s' % (repr(key),repr(value)))
    accessor_name = 'set' + UpperCase(key)
    if hasattr(aq_base(self),accessor_name):
      method = getattr(self, accessor_name)
      return method(value)
    else:
      setattr(self,key,value)
      #data_model = self.getTypeInfo().getDataModel(self)
      #type_info = self.getTypeInfo()
      #kw = {key:value}
      #type_info.editObject(self,kw)

  security.declarePrivate('edit' )
  def edit(self, REQUEST=None, force_update = 0, reindex_object = 0, **kw):
    return self._edit(REQUEST=REQUEST, force_update=force_update, reindex_object=reindex_object, **kw)


  # Object attributes update method
  security.declarePrivate( '_edit' )
  def _edit(self, REQUEST=None, force_update = 0, reindex_object = 0, **kw):
    """
      Generic edit Method for all ERP5 object
      The purpose of this method is to update attributed, eventually do
      some kind of type checking according to the property sheet and index
      the object.
  
      Each time attributes of an object are updated, they should
      be updated through this generic edit method
    """
    LOG('PatchCPSDoc._edit, kw: ',0,kw)
    try:
      categoryIds = self._getCategoryTool().getBaseCategoryIds()
    except:
      categoryIds = []
    #if kw.has_key('layout_and_schema'):
    #  self.setLayoutAndSchema(kw['layout_and_schema'])
    for key in kw.keys():
      accessor = 'get' + UpperCase(key)
      #if key in categoryIds:
      #  self._setCategoryMembership(key, kw[key])
      #if key != 'id' and key!= 'layout_and_schema':
      if key != 'id' :
        # We only change if the value is different
        # This may be very long.... 
        self._setProperty(key, kw[key])

def getCoverage(self):
  """
  """
  if hasattr(self,'coverage'):
    return self.coverage
  return None

def getCreator(self):
  """
  """
  #if hasattr(self,'coverage'):
  #  return self.coverage
  return None

def getRelation(self):
  """
  """
  if hasattr(self,'relation'):
    return self.relation
  return None

def getSource(self):
  """
  """
  if hasattr(self,'source'):
    return self.source
  return None

def getPreview(self):
  """
  """
  if hasattr(self,'preview'):
    return self.preview
  return None

def setCreator(self,value):
  """
  """
  setattr(self,'creator',value)

def setCreationDate(self,value):
  """
  """
  setattr(self,'creation_date',value)

CPSDocument.getCoverage = getCoverage
CPSDocument.getCreator = getCreator
CPSDocument.getRelation = getRelation
CPSDocument.setCreator = setCreator
CPSDocument.getSource = getSource
CPSDocument.getPreview = getPreview
CPSDocument.setCreationDate = setCreationDate
CPSDocument.getProperty = PatchedCPSDocument.getProperty
CPSDocument.getLayoutAndSchema = PatchedCPSDocument.getLayoutAndSchema
CPSDocument.setLayoutAndSchema = PatchedCPSDocument.setLayoutAndSchema
CPSDocument._propertyMap = PatchedCPSDocument._propertyMap
CPSDocument.setProperty = Base.setProperty
CPSDocument._setProperty = PatchedCPSDocument._setProperty
CPSDocument.asXML = Base.asXML
CPSDocument._edit = PatchedCPSDocument._edit
CPSDocument.edit = PatchedCPSDocument._edit