Telephone.py 9.54 KB
Newer Older
Jean-Paul Smets's avatar
Jean-Paul Smets committed
1 2 3
##############################################################################
#
# Copyright (c) 2002 Nexedi SARL and Contributors. All Rights Reserved.
4
#                    Jean-Paul Smets-Solanes <jp@nexedi.com>
Jean-Paul Smets's avatar
Jean-Paul Smets committed
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# 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 AccessControl import ClassSecurityInfo

from Products.ERP5Type import Permissions, PropertySheet, Constraint, Interface
from Products.ERP5Type.Base import Base

34
from Products.ERP5.Document.Coordinate import Coordinate
Jean-Paul Smets's avatar
Jean-Paul Smets committed
35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55

import re

class Telephone(Coordinate, Base):
    """
        A telephone is a coordinate which stores a telephone number
        The telephone class may be used by multiple content types (ex. Fax,
        Mobile Phone, Fax, Remote Access, etc.).

        A telephone is a terminating leaf
        in the OFS. It can not contain anything.

        Telephone inherits from Base and
        from the mix-in Coordinate

        A list of I18N telephone codes can be found here::
          http://kropla.com/dialcode.htm
    """

    meta_type = 'ERP5 Telephone'
    portal_type = 'Telephone'
56
    add_permission = Permissions.AddPortalContent
Jean-Paul Smets's avatar
Jean-Paul Smets committed
57 58 59 60 61
    isPortalContent = 1
    isRADContent = 1

    # Declarative security
    security = ClassSecurityInfo()
62
    security.declareObjectProtected(Permissions.AccessContentsInformation)
Jean-Paul Smets's avatar
Jean-Paul Smets committed
63 64 65 66 67 68 69

    # Declarative properties
    property_sheets = ( PropertySheet.Base
                      , PropertySheet.SimpleItem
                      , PropertySheet.Telephone
                      )

70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95
    #The notation always need to have:
    #<country> or <area> or <number> or <ext>
    #If uses a different tag it will be ignored.
    standard_dict={
      'default':{
        "input" : "((\+|)(?P<country>[\d]*))(0)?(( |)(\(0\)|)(\(|)(?P<area>[\d]*))?((\)|)(-|)(?P<number>[\d^ ^-]*))((\/|)(?P<ext>[\d]*))",
        "notation" : "+<country>(0)<area>-<number>/<ext>"
      },
      'france':{
        "input" : "((\+|)(?P<country>[\d]*))(0)?(( |)(\(0\)|)(\(|)(?P<area>[\d]*))?((\)|)(-|)(?P<number>[\d^ ^-]*))((\/|)(?P<ext>[\d]*))",
        "notation" : "+<country>(0)<area>-<number>/<ext>"
      },
      'brazil':{
        "input" : "((\+|)(?P<country>[\d]*))(0)?(( |)(\(0\)|)(\(|)(?P<area>[\d]*))?((\)|)(-|)(?P<number>[\d^ ^-]*))((\/|)(?P<ext>[\d]*))",
        "notation" : "+<country>(<area>)<number>/<ext>",
      },
      'dakar':{
        "input" : "((\+|)(\(|)(?P<country>[\d]*))?((\)|)(-|)(?P<number>[\d^ ^-]*))((\/|)(?P<ext>[\d]*)",
        "notation" : "+<country> <number>/<ext>",
      },
      'tokio':{
        "input" : "((\+|)(?P<country>[\d]*))(0)?(( |)(\(0\)|)(\(|)(?P<area>[\d]*))?((\)|)(-|)(?P<number>[\d^ ^-]*))((\/|)(?P<ext>[\d]*))",
        "notation" : "+<country>(<area>)<number>/<ext>",
      }
    }
    
Jean-Paul Smets's avatar
Jean-Paul Smets committed
96
    security.declareProtected(Permissions.ModifyPortalContent, 'fromText')
97
    def fromText(self, coordinate_text):
98 99 100 101 102 103 104 105 106 107 108 109 110 111
      """ See ICoordinate.fromText """
      method = self._getTypeBasedMethod('fromText')
      if method is not None:
        return method(text=coordinate_text)
      
      if coordinate_text is None:
        coordinate_text = ''

      #This regexp get the coordinate text and extract only numbers
      onlynumber = ''.join(re.findall('[0-9]', coordinate_text))
      
      #Test if coordinate_text has or not markups.
      if len(coordinate_text) > len(onlynumber):
        #trying to get a possible contry number to be used by script
112 113 114 115 116 117 118 119
        country=re.match('((\+|)(?P<country>\d*))', \
                         coordinate_text).groupdict().get('country','')
        regexdict = self.getRegexDict(index=country)
        input_parser = regexdict['input']
        number_match = re.match(input_parser, coordinate_text)
        if not number_match:
          return
        number_dict = number_match.groupdict()
120 121 122 123 124 125 126 127 128 129 130 131
      else:
        number_dict={'number':coordinate_text}
      
      country=number_dict.get('country','')
      area=number_dict.get('area','')
      number=number_dict.get('number','')
      ext=number_dict.get('ext','')

      if ((country in ['', None]) and \
          (area in ['', None]) and \
          (number in ['', None]) and \
          (ext in ['', None])):
132
        country = area = number = extension=''
133 134 135 136
      else:
        #The country and area is trying to get from dict, 
        #but if it fails must be get from preference
        country = (number_dict.get('country') or \
137
                   self.portal_preferences.getPreferredTelephoneDefaultCountryNumber() or \
138 139
                   '').strip()
        area = (number_dict.get('area') or \
140
                self.portal_preferences.getPreferredTelephoneDefaultAreaNumber() or 
141 142 143 144 145 146 147 148 149
                '').strip()
        number = (number_dict.get('number') or '').strip().replace('-', '').replace(' ','')
        extension = (number_dict.get('ext') or '').strip()

      self.edit(telephone_country = country,
                telephone_area = area,
                telephone_number = number, 
                telephone_extension = extension)
     
150 151
    security.declareProtected(Permissions.ModifyPortalContent, '_setText')
    _setText = fromText
Jean-Paul Smets's avatar
Jean-Paul Smets committed
152 153 154

    security.declareProtected(Permissions.View, 'asText')
    def asText(self):
155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174
      """
        Returns the telephone number in standard format
      """
      script = self._getTypeBasedMethod('asText')
      if script is not None:
        return script()
      
      telephone_country = self.getTelephoneCountry() or ''
      telephone_area = self.getTelephoneArea() or ''
      telephone_number = self.getTelephoneNumber() or ''
      telephone_extension = self.getTelephoneExtension() or ''
     
      # If country, area, number, extension are blank the method
      # should to return blank.
      if ((telephone_country == '') and \
          (telephone_area == '') and \
          (telephone_number == '') and \
          (telephone_extension == '')): 
        return ''
      
175 176
      regexdict = self.getRegexDict(index=telephone_country)
      notation = regexdict['notation']
177 178 179 180 181 182 183 184

      if notation not in [None, '']:
        notation=notation.replace('<country>',telephone_country)
        notation=notation.replace('<area>',telephone_area)
        notation=notation.replace('<number>',telephone_number)
        notation=notation.replace('<ext>',telephone_extension)

      return notation
Jean-Paul Smets's avatar
Jean-Paul Smets committed
185

186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209
    security.declareProtected(Permissions.AccessContentsInformation,
                              'asURL')
    def asURL(self):
      """Returns a text representation of the Url if defined
      or None else.
      """
      telephone_country = self.getTelephoneCountry()
      if telephone_country is not None:
        url_string = '+%s' % telephone_country
      else :
        url_string = '0'

      telephone_area = self.getTelephoneArea()
      if telephone_area is not None:
        url_string += telephone_area

      telephone_number = self.getTelephoneNumber()
      if telephone_number is not None:
        url_string += telephone_number

      if url_string == '0':
        return None
      return 'tel:%s' % (url_string.replace(' ',''))

210 211
    security.declareProtected(Permissions.View, 'getText')
    getText = asText
Jean-Paul Smets's avatar
Jean-Paul Smets committed
212 213 214

    security.declareProtected(Permissions.View, 'standardTextFormat')
    def standardTextFormat(self):
215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
      """
        Returns the standard text formats for telephone numbers
      """
      return ("+33(0)6-62 05 76 14",)

    def getRegexDict(self, index=None):
      """
        Returns a dict with Regex and Notations based from 
        country or region
      """
      # In case of index is a number
      # should return a region from dict or from form field
      if index not in [None,'']:
        countrydict = {
          '33':'france',
          '55':'brazil',
          '221':'dakar',
          '64':'tokio'
        }
        #If index is not in country list
        #the possible country can be found at region field
        if index not in countrydict.keys():
          region = self.getRegion() or \
             self.portal_preferences.getPreferredTelephoneDefaultRegion()
        else:
          region = countrydict.get(index,'')
      else:
        region = self.getRegion() or \
          self.portal_preferences.getPreferredTelephoneDefaultRegion()
      
      # Find the region in regexdict
      if region is not None:
        regionlist=region.split('/')
        for i in regionlist:
          if self.standard_dict.get(i) is not None:
            region = i
      
      # If region doesn't exist the script should to return a default regex.
      if region not in self.standard_dict.keys() or region is None:
        region = 'default'
      
      dict = self.standard_dict.get(region)
      return dict
Jean-Paul Smets's avatar
Jean-Paul Smets committed
258