'unknown', self::GENDER_MALE => 'male', self::GENDER_FEMALE => 'female', ]; /** * Phone types. */ const PHONE_TYPE_WORK = 1; const PHONE_TYPE_HOME = 2; const PHONE_TYPE_OTHER = 3; const PHONE_TYPE_MOBILE = 4; const PHONE_TYPE_FAX = 5; const PHONE_TYPE_PAGER = 6; /** * For API. */ public static $phone_types = [ self::PHONE_TYPE_WORK => 'work', self::PHONE_TYPE_HOME => 'home', self::PHONE_TYPE_MOBILE => 'mobile', self::PHONE_TYPE_FAX => 'fax', self::PHONE_TYPE_PAGER => 'pager', self::PHONE_TYPE_OTHER => 'other', ]; /** * Photo types. */ const PHOTO_TYPE_UKNOWN = 1; const PHOTO_TYPE_GRAVATAR = 2; const PHOTO_TYPE_TWITTER = 3; const PHOTO_TYPE_FACEBOOK = 4; const PHOTO_TYPE_GOOGLEPROFILE = 5; const PHOTO_TYPE_GOOGLEPLUS = 6; const PHOTO_TYPE_LINKEDIN = 7; const PHOTO_TYPE_VK = 8; // Extra /** * For API. */ public static $photo_types = [ self::PHOTO_TYPE_UKNOWN => 'unknown', self::PHOTO_TYPE_GRAVATAR => 'gravatar', self::PHOTO_TYPE_TWITTER => 'twitter', self::PHOTO_TYPE_FACEBOOK => 'facebook', self::PHOTO_TYPE_GOOGLEPROFILE => 'googleprofile', self::PHOTO_TYPE_GOOGLEPLUS => 'googleplus', self::PHOTO_TYPE_LINKEDIN => 'linkedin', self::PHOTO_TYPE_VK => 'vk', // Extra ]; /** * Chat types. */ // const CHAT_TYPE_AIM = 1; // const CHAT_TYPE_GTALK = 2; // const CHAT_TYPE_ICQ = 3; // const CHAT_TYPE_XMPP = 4; // const CHAT_TYPE_MSN = 5; // const CHAT_TYPE_SKYPE = 6; // const CHAT_TYPE_YAHOO = 7; // const CHAT_TYPE_QQ = 8; // const CHAT_TYPE_WECHAT = 10; // const CHAT_TYPE_OTHER = 9; /** * For API. */ // public static $chat_types = [ // self::CHAT_TYPE_AIM => 'aim', // self::CHAT_TYPE_GTALK => 'gtalk', // self::CHAT_TYPE_ICQ => 'icq', // self::CHAT_TYPE_XMPP => 'xmpp', // self::CHAT_TYPE_MSN => 'msn', // self::CHAT_TYPE_SKYPE => 'skype', // self::CHAT_TYPE_YAHOO => 'yahoo', // self::CHAT_TYPE_QQ => 'qq', // self::CHAT_TYPE_WECHAT => 'wechat', // Extra // self::CHAT_TYPE_OTHER => 'other', // ]; // public static $chat_type_names = [ // self::CHAT_TYPE_AIM => 'AIM', // self::CHAT_TYPE_GTALK => 'Google+', // self::CHAT_TYPE_ICQ => 'ICQ', // self::CHAT_TYPE_XMPP => 'XMPP', // self::CHAT_TYPE_MSN => 'MSN', // self::CHAT_TYPE_SKYPE => 'Skype', // self::CHAT_TYPE_YAHOO => 'Yahoo', // self::CHAT_TYPE_QQ => 'QQ', // self::CHAT_TYPE_WECHAT => 'WeChat', // Extra // self::CHAT_TYPE_OTHER => 'Other', // ]; /** * Social types. */ const SOCIAL_TYPE_TWITTER = 1; const SOCIAL_TYPE_FACEBOOK = 2; const SOCIAL_TYPE_TELEGRAM = 14; const SOCIAL_TYPE_LINKEDIN = 3; const SOCIAL_TYPE_ABOUTME = 4; const SOCIAL_TYPE_GOOGLE = 5; const SOCIAL_TYPE_GOOGLEPLUS = 6; const SOCIAL_TYPE_TUNGLEME = 7; const SOCIAL_TYPE_QUORA = 8; const SOCIAL_TYPE_FOURSQUARE = 9; const SOCIAL_TYPE_YOUTUBE = 10; const SOCIAL_TYPE_FLICKR = 11; const SOCIAL_TYPE_VK = 13; // Extra const SOCIAL_TYPE_OTHER = 12; public static $social_types = [ self::SOCIAL_TYPE_TWITTER => 'twitter', self::SOCIAL_TYPE_FACEBOOK => 'facebook', self::SOCIAL_TYPE_TELEGRAM => 'telegram', self::SOCIAL_TYPE_LINKEDIN => 'linkedin', self::SOCIAL_TYPE_ABOUTME => 'aboutme', self::SOCIAL_TYPE_GOOGLE => 'google', self::SOCIAL_TYPE_GOOGLEPLUS => 'googleplus', self::SOCIAL_TYPE_TUNGLEME => 'tungleme', self::SOCIAL_TYPE_QUORA => 'quora', self::SOCIAL_TYPE_FOURSQUARE => 'foursquare', self::SOCIAL_TYPE_YOUTUBE => 'youtube', self::SOCIAL_TYPE_FLICKR => 'flickr', self::SOCIAL_TYPE_VK => 'vk', // Extra self::SOCIAL_TYPE_OTHER => 'other', ]; public static $social_type_names = [ self::SOCIAL_TYPE_TWITTER => 'Twitter', self::SOCIAL_TYPE_FACEBOOK => 'Facebook', self::SOCIAL_TYPE_TELEGRAM => 'Telegram', self::SOCIAL_TYPE_LINKEDIN => 'Linkedin', self::SOCIAL_TYPE_ABOUTME => 'About.me', self::SOCIAL_TYPE_GOOGLE => 'Google', self::SOCIAL_TYPE_GOOGLEPLUS => 'Google+', self::SOCIAL_TYPE_TUNGLEME => 'Tungle.me', self::SOCIAL_TYPE_QUORA => 'Quora', self::SOCIAL_TYPE_FOURSQUARE => 'Foursquare', self::SOCIAL_TYPE_YOUTUBE => 'YouTube', self::SOCIAL_TYPE_FLICKR => 'Flickr', self::SOCIAL_TYPE_VK => 'VK', self::SOCIAL_TYPE_OTHER => 'Other', ]; /** * Search filters. */ public static $search_filters = [ 'mailbox', ]; /** * Countries list. */ public static $countries = [ 'US' => 'United States', 'AU' => 'Australia', 'CA' => 'Canada', 'DK' => 'Denmark', 'FR' => 'France', 'DE' => 'Germany', 'IT' => 'Italy', 'JP' => 'Japan', 'MX' => 'Mexico', 'ES' => 'Spain', 'SE' => 'Sweden', 'GB' => 'United Kingdom', 'AF' => 'Afghanistan', 'AX' => 'Ă…land Islands', 'AL' => 'Albania', 'DZ' => 'Algeria', 'AS' => 'American Samoa', 'AD' => 'Andorra', 'AO' => 'Angola', 'AI' => 'Anguilla', 'AQ' => 'Antarctica', 'AG' => 'Antigua and Barbuda', 'AR' => 'Argentina', 'AM' => 'Armenia', 'AW' => 'Aruba', 'AT' => 'Austria', 'AZ' => 'Azerbaijan', 'BS' => 'Bahamas', 'BH' => 'Bahrain', 'BD' => 'Bangladesh', 'BB' => 'Barbados', 'BY' => 'Belarus', 'BE' => 'Belgium', 'BZ' => 'Belize', 'BJ' => 'Benin', 'BM' => 'Bermuda', 'BT' => 'Bhutan', 'BO' => 'Bolivia', 'BQ' => 'Bonaire', 'BA' => 'Bosnia and Herzegowina', 'BW' => 'Botswana', 'BV' => 'Bouvet Island', 'BR' => 'Brazil', 'IO' => 'British Indian Ocean Territory', 'BN' => 'Brunei Darussalam', 'BG' => 'Bulgaria', 'BF' => 'Burkina Faso', 'BI' => 'Burundi', 'KH' => 'Cambodia', 'CM' => 'Cameroon', 'CV' => 'Cape Verde', 'KY' => 'Cayman Islands', 'CF' => 'Central African Republic', 'TD' => 'Chad', 'CL' => 'Chile', 'CN' => 'China', 'CX' => 'Christmas Island', 'CC' => 'Cocos (Keeling) Islands', 'CO' => 'Colombia', 'KM' => 'Comoros', 'CG' => 'Congo', 'CD' => 'Congo, DR', 'CK' => 'Cook Islands', 'CR' => 'Costa Rica', 'CI' => "Cote D'Ivoire", 'HR' => 'Croatia', 'CU' => 'Cuba', 'CW' => 'Curacao', 'CY' => 'Cyprus', 'CZ' => 'Czech Republic', 'DJ' => 'Djibouti', 'DM' => 'Dominica', 'DO' => 'Dominican Republic', 'EC' => 'Ecuador', 'EG' => 'Egypt', 'SV' => 'El Salvador', 'GQ' => 'Equatorial Guinea', 'ER' => 'Eritrea', 'EE' => 'Estonia', 'ET' => 'Ethiopia', 'FK' => 'Falkland Islands (Malvinas)', 'FO' => 'Faroe Islands', 'FJ' => 'Fiji', 'FI' => 'Finland', 'GF' => 'French Guiana', 'PF' => 'French Polynesia', 'TF' => 'French Southern Territories', 'GA' => 'Gabon', 'GM' => 'Gambia', 'GE' => 'Georgia', 'GH' => 'Ghana', 'GI' => 'Gibraltar', 'GR' => 'Greece', 'GL' => 'Greenland', 'GD' => 'Grenada', 'GP' => 'Guadeloupe', 'GU' => 'Guam', 'GT' => 'Guatemala', 'GG' => 'Guernsey', 'GN' => 'Guinea', 'GW' => 'Guinea-bissau', 'GY' => 'Guyana', 'HT' => 'Haiti', 'HM' => 'Heard and Mc Donald Islands', 'HN' => 'Honduras', 'HK' => 'Hong Kong', 'HU' => 'Hungary', 'IS' => 'Iceland', 'IN' => 'India', 'ID' => 'Indonesia', 'IR' => 'Iran (Islamic Republic of)', 'IQ' => 'Iraq', 'IE' => 'Ireland', 'IM' => 'Isle of Man', 'IL' => 'Israel', 'JM' => 'Jamaica', 'JE' => 'Jersey', 'JO' => 'Jordan', 'KZ' => 'Kazakhstan', 'KE' => 'Kenya', 'KI' => 'Kiribati', 'KP' => "Korea, Democratic People's Republic of", 'KR' => 'Korea, Republic of', 'KW' => 'Kuwait', 'KG' => 'Kyrgyzstan', 'LA' => "Lao People's Democratic Republic", 'LV' => 'Latvia', 'LB' => 'Lebanon', 'LS' => 'Lesotho', 'LR' => 'Liberia', 'LY' => 'Libya', 'LI' => 'Liechtenstein', 'LT' => 'Lithuania', 'LU' => 'Luxembourg', 'MO' => 'Macao', 'MK' => 'Macedonia, The Former Yugoslav Republic of', 'MG' => 'Madagascar', 'MW' => 'Malawi', 'MY' => 'Malaysia', 'MV' => 'Maldives', 'ML' => 'Mali', 'MT' => 'Malta', 'MH' => 'Marshall Islands', 'MQ' => 'Martinique', 'MR' => 'Mauritania', 'MU' => 'Mauritius', 'YT' => 'Mayotte', 'FM' => 'Micronesia, Federated States of', 'MD' => 'Moldova, Republic of', 'MC' => 'Monaco', 'MN' => 'Mongolia', 'ME' => 'Montenegro', 'MS' => 'Montserrat', 'MA' => 'Morocco', 'MZ' => 'Mozambique', 'MM' => 'Myanmar', 'NA' => 'Namibia', 'NR' => 'Nauru', 'NP' => 'Nepal', 'NL' => 'Netherlands', 'NC' => 'New Caledonia', 'NZ' => 'New Zealand', 'NI' => 'Nicaragua', 'NE' => 'Niger', 'NG' => 'Nigeria', 'NU' => 'Niue', '00' => 'None Available', 'NF' => 'Norfolk Island', 'MP' => 'Northern Mariana Islands', 'NO' => 'Norway', 'OM' => 'Oman', 'PK' => 'Pakistan', 'PW' => 'Palau', 'PS' => 'Palestine, State of', 'PA' => 'Panama', 'PG' => 'Papua New Guinea', 'PY' => 'Paraguay', 'PE' => 'Peru', 'PH' => 'Philippines', 'PN' => 'Pitcairn', 'PL' => 'Poland', 'PT' => 'Portugal', 'PR' => 'Puerto Rico', 'QA' => 'Qatar', 'RE' => 'Reunion', 'RO' => 'Romania', 'RU' => 'Russia', 'RW' => 'Rwanda', 'BL' => 'Saint Barthelemy', 'KN' => 'Saint Kitts and Nevis', 'LC' => 'Saint Lucia', 'MF' => 'Saint Martin (French part)', 'VC' => 'Saint Vincent and the Grenadines', 'WS' => 'Samoa', 'SM' => 'San Marino', 'ST' => 'Sao Tome and Principe', 'SA' => 'Saudi Arabia', 'SN' => 'Senegal', 'RS' => 'Serbia', 'SC' => 'Seychelles', 'SL' => 'Sierra Leone', 'SG' => 'Singapore', 'SX' => 'Sint Maarten (Dutch part)', 'SK' => 'Slovakia', 'SI' => 'Slovenia', 'SB' => 'Solomon Islands', 'SO' => 'Somalia', 'ZA' => 'South Africa', 'GS' => 'South Georgia and the South Sandwich Islands', 'SS' => 'South Sudan', 'LK' => 'Sri Lanka', 'SH' => 'St. Helena', 'PM' => 'St. Pierre and Miquelon', 'SD' => 'Sudan', 'SR' => 'Suriname', 'SJ' => 'Svalbard and Jan Mayen', 'SZ' => 'Swaziland', 'CH' => 'Switzerland', 'SY' => 'Syrian Arab Republic', 'TW' => 'Taiwan, Province of China', 'TJ' => 'Tajikistan', 'TZ' => 'Tanzania, United Republic of', 'TH' => 'Thailand', 'TL' => 'Timor-Leste', 'TG' => 'Togo', 'TK' => 'Tokelau', 'TO' => 'Tonga', 'TT' => 'Trinidad and Tobago', 'TN' => 'Tunisia', 'TR' => 'Turkey', 'TM' => 'Turkmenistan', 'TC' => 'Turks and Caicos Islands', 'TV' => 'Tuvalu', 'UG' => 'Uganda', 'UA' => 'Ukraine', 'AE' => 'United Arab Emirates', 'UM' => 'United States Minor Outlying Islands', 'UY' => 'Uruguay', 'UZ' => 'Uzbekistan', 'VU' => 'Vanuatu', 'VA' => 'Vatican City State (Holy See)', 'VE' => 'Venezuela', 'VN' => 'Vietnam', 'VG' => 'Virgin Islands (British)', 'VI' => 'Virgin Islands (U.S.)', 'WF' => 'Wallis and Futuna Islands', 'EH' => 'Western Sahara', 'YE' => 'Yemen', 'ZM' => 'Zambia', 'ZW' => 'Zimbabwe', ]; protected $casts = [ 'meta' => 'array', ]; /** * Attributes which are not fillable using fill() method. */ protected $guarded = ['id']; /** * Attributes fillable using fill() method. * * @var [type] */ protected $fillable = ['first_name', 'last_name', 'company', 'job_title', 'address', 'city', 'state', 'zip', 'country', 'photo_url', 'age', 'gender', 'notes', 'channel', 'channel_id', 'social_profiles']; /** * Fields stored as JSON. */ protected $json_fields = ['phones', 'websites', 'social_profiles']; /** * Get customer emails. */ public function emails() { return $this->hasMany('App\Email'); } /** * Get customer emails. */ public function emails_cached() { return $this->hasMany('App\Email')->rememberForever(); } /** * Get customer conversations. */ public function conversations() { return $this->hasMany('App\Conversation'); } /** * Get main email. */ public function getMainEmail() { return optional($this->emails_cached()->first())->email.''; } /** * Get main email. */ public static function getMainEmailStatic($customer_id) { return Email::select('email')->where('customer_id', $customer_id)->pluck('email'); } /** * Get customer full name. * * @return string */ public function getFullName($email_if_empty = false, $first_part_from_email = false) { if ($this->first_name && $this->last_name) { return $this->first_name.' '.$this->last_name; } elseif (!$this->last_name && $this->first_name) { return $this->first_name; } elseif (!$this->first_name && $this->last_name) { return $this->last_name; } elseif ($email_if_empty) { $email = $this->getMainEmail(); if ($first_part_from_email) { return $this->getNameFromEmail($email); } else { return $email; } } return ''; } /** * Get customer first name. * * @return string */ public function getFirstName($email_if_empty = false) { if ($this->first_name) { return $this->first_name; } elseif ($email_if_empty) { return $this->getNameFromEmail(); } return ''; } /** * Get first part of the email. * * @return string */ public function getNameFromEmail($email = '') { if (!$email) { $email = optional($this->emails_cached()->first())->email; } if ($email) { return explode('@', $email)[0]; } else { return ''; } } /** * Set customer emails. * * @param array $emails */ public function syncEmails($emails) { if (is_array($emails)) { $deleted_emails = []; foreach ($this->emails as $email) { foreach ($emails as $email_address) { if (Email::sanitizeEmail($email->email) == Email::sanitizeEmail($email_address)) { continue 2; } } $deleted_emails[] = $email; } foreach ($emails as $email_address) { $email_address = Email::sanitizeEmail($email_address); if (!$email_address) { continue; } $email = Email::where('email', $email_address)->first(); $new_emails = []; if ($email) { // Assign email to current customer if ($email->customer_id != $this->id) { $email->customer()->associate($this); $email->save(); } } else { $new_emails[] = new Email(['email' => $email_address]); } if ($new_emails) { $this->emails()->saveMany($new_emails); } } foreach ($deleted_emails as $email) { if (Conversation::where('customer_email', $email->email)->exists()) { // Create customers for deleted emails // if there is a conversation with 'customer_email'. $customer = new self(); $customer->save(); $email->customer()->associate($customer); $email->save(); } else { // Simply delete an email. $email->delete(); } } } } /** * Add new email to customer. */ public function addEmail($email_address, $check_if_exists = false) { // Check if email already exists and belongs to another customer. if ($check_if_exists) { $email = Email::where('email', $email_address)->first(); if ($email && !empty($email->customer_id)) { return false; } } $new_email = new Email(['email' => $email_address]); $this->emails()->save($new_email); } /** * Get customers phones as array. * * @return array */ public function getPhones($dummy_if_empty = false) { $phones = json_decode($this->phones ?? '', true); if (is_array($phones) && count($phones)) { return $phones; } elseif ($dummy_if_empty) { return [[ 'value' => '', 'type' => self::PHONE_TYPE_WORK, ]]; } else { return []; } } public function getMainPhoneValue() { return $this->getMainPhoneNumber(); } public function getMainPhoneNumber() { $phones = $this->getPhones(); return $phones[0]['value'] ?? ''; } /** * Set phones as JSON. * * @param array $phones_array */ public function setPhones(array $phones_array) { $phones_array = self::formatPhones($phones_array); // Remove dubplicates. $list = []; foreach ($phones_array as $i => $data) { if (in_array($data['value'], $list)) { unset($phones_array[$i]); } else { $list[] = $data['value']; } } $this->phones = \Helper::jsonEncodeUtf8($phones_array); } /** * Sanitize phones array. * * @param array $phones_array [description] * * @return array [description] */ public static function formatPhones(array $phones_array) { $phones = []; foreach ($phones_array as $phone) { if (is_array($phone)) { if (!empty($phone['value'])) { if (empty($phone['type']) || !in_array($phone['type'], array_keys(self::$phone_types))) { $phone['type'] = self::PHONE_TYPE_WORK; } $phones[] = [ 'value' => (string) $phone['value'], 'type' => (int) $phone['type'], 'n' => (string)\Helper::phoneToNumeric($phone['value']), ]; } } else { $phones[] = [ 'value' => (string) $phone, 'type' => self::PHONE_TYPE_WORK, 'n' => (string)\Helper::phoneToNumeric($phone), ]; } } return $phones; } /** * Add website. */ public function addPhone($phone, $type = self::PHONE_TYPE_WORK) { if (is_string($phone)) { $this->setPhones(array_merge( $this->getPhones(), [['value' => $phone, 'type' => $type]] )); } else { $this->setPhones(array_merge( $this->getPhones(), [$phone] )); } } /** * Find customer by phone number. */ public static function findByPhone($phone) { return Customer::byPhone($phone)->first(); } /** * Get query. */ public static function byPhone($phone) { $phone_numeric = \Helper::phoneToNumeric($phone); return Customer::where('phones', 'LIKE', '%"'.$phone_numeric.'"%'); } /** * Get customers social profiles as array. * * @return array */ public function getSocialProfiles($dummy_if_empty = false) { $social_profiles = json_decode($this->social_profiles ?? '', true); if (is_array($social_profiles) && count($social_profiles)) { return json_decode($this->social_profiles, true); } elseif ($dummy_if_empty) { return [[ 'type' => '', 'value' => '', ]]; } else { return []; } } /** * Get customers social profiles as array. * * @return array */ public function getWebsites($dummy_if_empty = false) { $websites = json_decode($this->websites ?? '', true); if (is_array($websites) && count($websites)) { return $websites; } elseif ($dummy_if_empty) { return ['']; } else { return []; } } public function getMainWebsite() { $websites = $this->getWebsites(); return $websites[0] ?? ''; } /** * Set websites as JSON. * * @param array $websites_array */ public function setWebsites(array $websites_array) { $websites = []; foreach ($websites_array as $key => $value) { // FILTER_SANITIZE_URL cuts some symbols. //$value = filter_var((string) $value, FILTER_SANITIZE_URL); if (isset($value['value'])) { $value = $value['value']; } if (!$value || preg_match("/^http(s)?:?\/?\/?$/i", $value)) { continue; } if (!preg_match("/http(s)?:\/\//i", $value)) { $value = 'http://'.$value; } $websites[] = (string) $value; } $this->websites = \Helper::jsonEncodeUtf8(array_unique($websites)); } /** * Add website. */ public function addWebsite($website) { $websites = $this->getWebsites(); if (isset($website['value'])) { $website = $website['value']; } array_push($websites, $website); $this->setWebsites($websites); } /** * Sanitize social profiles. * * @param array $list [description] * * @return array [description] */ public static function formatSocialProfiles(array $list) { $social_profiles = []; foreach ($list as $social_profile) { if (is_array($social_profile)) { if (!empty($social_profile['value']) && !empty($social_profile['type'])) { $type = null; if (is_numeric($social_profile['type']) && in_array($social_profile['type'], array_keys(self::$social_types))) { $type = (int)$social_profile['type']; } else { // Find type. foreach (self::$social_types as $type_id => $type_name) { if ($type_name == strtolower($social_profile['type'])) { $type = $type_id; } } } if (!$type) { continue; } $social_profiles[] = [ // Order of elements in array is important as we rely on it // when searching customers by social profiles using "like". 'value' => (string) $social_profile['value'], 'type' => $type, ]; } } else { $social_profiles[] = [ // Order of elements in array is important as we rely on it // when searching customers by social profiles using "like". 'value' => (string) $social_profile, 'type' => self::SOCIAL_TYPE_OTHER, ]; } } return $social_profiles; } /** * Set social profiles as JSON. * * @param array $websites_array */ public function setSocialProfiles(array $sp_array) { $sp_array = self::formatSocialProfiles($sp_array); // Remove dubplicates. $list = []; foreach ($sp_array as $i => $data) { if (in_array($data['value'], $list)) { unset($sp_array[$i]); } else { $list[] = $data['value']; } } $this->social_profiles = \Helper::jsonEncodeUtf8($sp_array); } /** * Create customer or get existing and fill empty fields. * * @param string $email * @param array $data [description] * * @return [type] [description] */ public static function create($email, $data = []) { $new = false; $email = Email::sanitizeEmail($email); if (!$email) { return null; } $email_obj = Email::where('email', $email)->first(); if ($email_obj) { $customer = $email_obj->customer; // In case somehow the email has no customer. if (!$customer) { // Customer will be saved and connected to the email later. $customer = new self(); } // Update name if empty. /*if (empty($customer->first_name) && !empty($data['first_name'])) { $customer->first_name = $data['first_name']; if (empty($customer->last_name) && !empty($data['last_name'])) { $customer->last_name = $data['last_name']; } $customer->save(); }*/ } else { $customer = new self(); $email_obj = new Email(); $email_obj->email = $email; $new = true; } // Set empty fields if ($customer->setData($data, false) || !$customer->id) { $customer->save(); } if (empty($email_obj->id) || !$email_obj->customer_id || $email_obj->customer_id != $customer->id) { // Email may have been set in setData(). $save_email = true; if (!empty($data['emails']) && is_array($data['emails'])) { foreach ($data['emails'] as $data_email) { if (is_string($data_email) && $data_email == $email) { $save_email = false; break; } if (is_array($data_email) && !empty($data_email['value']) && $data_email['value'] == $email) { $save_email = false; break; } } } if ($save_email) { $email_obj->customer()->associate($customer); $email_obj->save(); } } // Todo: check phone uniqueness. if ($new) { \Eventy::action('customer.created', $customer); } return $customer; } /** * Set empty fields. */ public function setData($data, $replace_data = true, $save = false) { $result = false; // todo: photoUrl. if (isset($data['photo_url'])) { unset($data['photo_url']); } if (!empty($data['background']) && empty($data['notes'])) { $data['notes'] = $data['background']; } if ($replace_data) { // Replace data. $data_prepared = $data; foreach ($data_prepared as $i => $value) { if (is_array($value)) { unset($data_prepared[$i]); } } $this->fill($data_prepared); $result = true; } else { // Update empty fields. foreach ($data as $key => $value) { if (in_array($key, $this->fillable) && empty($this->$key)) { $this->$key = $value; $result = true; } } } // Set JSON values. if (!empty($data['phone'])) { $this->addPhone($data['phone']); } foreach ($data as $key => $value) { if (!in_array($key, $this->json_fields) && $key != 'emails') { continue; } if ($key == 'phones') { if (isset($value['value'])) { $this->addPhone($value); } else { $this->setPhones($value); // foreach ($value as $phone_value) { // $this->addPhone($phone_value); // } } $result = true; } if ($key == 'websites') { if (is_array($value)) { $this->setWebsites($value); // foreach ($value as $website) { // $this->addWebsite($website); // } } else { $this->addWebsite($value); } $result = true; } if ($key == 'social_profiles') { $this->setSocialProfiles($value); $result = true; } if ($key == 'country') { if (array_search($this->country, Customer::$countries)) { $this->country = array_search($this->country, Customer::$countries); } $this->country = strtoupper(mb_substr($this->country, 0, 2)); $result = true; } } // Emails must be processed the last as they need to save object. foreach ($data as $key => $value) { if ($key == 'emails') { foreach ($value as $email_data) { if (is_string($email_data)) { if (!$this->id) { $this->save(); } $email_created = Email::create($email_data, $this->id, Email::TYPE_WORK); if ($email_created) { $result = true; } } elseif (!empty($email_data['value'])) { if (!$this->id) { $this->save(); } $email_created = Email::create($email_data['value'], $this->id, $email_data['type']); if ($email_created) { $result = true; } } } break; } } // Maybe Todo: check phone uniqueness. // Same phone can be written in many ways, so it's almost useless to chek uniqueness. \Eventy::action('customer.set_data', $this, $data, $replace_data); if ($save) { $this->save(); } return $result; } /** * Create a customer, email is not required. * For phone conversations. */ public static function createWithoutEmail($data = []) { $customer = new self(); $customer->setData($data); $customer->save(); \Eventy::action('customer.created', $customer); return $customer; } /** * Get customer URL. * * @return string */ public function url() { return route('customers.update', ['id'=>$this->id]); } /** * Get view customer URL. * * @return string */ public function urlView() { return route('customers.conversations', ['id'=>$this->id]); } /** * Format date according to customer's timezone. * * @param Carbon $date * @param string $format * * @return string */ public static function dateFormat($date, $format = 'M j, Y H:i') { return $date->format($format); } /** * Get full representation of customer. */ public function getEmailAndName() { // Email can be fetched using query. $text = $this->email; if (!$text) { $text = $this->getMainEmail(); } if ($this->getFullName()) { if ($text) { $text .= ' ('.$this->getFullName().')'; } else { $text .= $this->getFullName(); } } return $text; } public function getNameAndEmail() { // Email can be fetched using query. $text = $this->getFullName(); $email = $this->email; if (!$email) { $email = $this->getMainEmail(); } if ($email) { if ($text) { $text .= ' <'.$email.'>'; } else { $text .= $email; } } return $text; } /** * Get customers info for the list of emails. */ public static function emailsToCustomers($list) { $result = []; $data = Customer::select(['emails.email', 'customers.first_name', 'customers.last_name']) ->join('emails', function ($join) { $join->on('emails.customer_id', '=', 'customers.id'); }) ->whereIn('emails.email', $list) //->groupby('customers.id') ->get() ->toArray(); foreach ($list as $email) { // Dummy customer. $customer = new Customer(); $customer->email = $email; foreach ($data as $values) { if (strtolower($values['email']) == strtolower($email)) { $customer->first_name = $values['first_name']; $customer->last_name = $values['last_name']; break; } } $result[$email] = $customer->getNameAndEmail(); } return $result; } /** * Get customer by email. */ public static function getByEmail($email) { return Customer::select('customers.*') ->where('emails.email', $email) ->join('emails', function ($join) { $join->on('emails.customer_id', '=', 'customers.id'); })->first(); } /** * Get email or phone if email is empty. */ public function getEmailOrPhone() { if (!empty($this->email)) { // Email can be selected with query. return $this->email; } elseif ($main_email = $this->getMainEmail()) { return $main_email; } elseif ($phones = $this->getPhones() && !empty($phones[0]['value'])) { return $phones[0]['value']; } return ''; } public function getPhotoUrl($default_if_empty = true) { if (!empty($this->photo_url) || !$default_if_empty) { if (!empty($this->photo_url)) { return self::getPhotoUrlByFileName($this->photo_url); } else { return ''; } } else { return \Eventy::filter('customer.default_avatar', asset('/img/default-avatar.png'), $this); } } public static function getPhotoUrlByFileName($file_name) { return Storage::url(self::PHOTO_DIRECTORY.DIRECTORY_SEPARATOR.$file_name); } /** * Resize and save photo. */ public function savePhoto($real_path, $mime_type) { $photo_size = config('app.customer_photo_size'); $resized_image = \App\Misc\Helper::resizeImage($real_path, $mime_type, $photo_size, $photo_size); if (!$resized_image) { return false; } $file_name = md5(Hash::make($this->id)).'.jpg'; $dest_path = Storage::path(self::PHOTO_DIRECTORY.DIRECTORY_SEPARATOR.$file_name); $dest_dir = pathinfo($dest_path, PATHINFO_DIRNAME); if (!file_exists($dest_dir)) { \File::makeDirectory($dest_dir, \Helper::DIR_PERMISSIONS); } // Remove current photo if ($this->photo_url) { Storage::delete(self::PHOTO_DIRECTORY.DIRECTORY_SEPARATOR.$this->photo_url); } imagejpeg($resized_image, $dest_path, self::PHOTO_QUALITY); return $file_name; } /** * Remove user photo. */ public function removePhoto() { if ($this->photo_url) { Storage::delete(self::PHOTO_DIRECTORY.DIRECTORY_SEPARATOR.$this->photo_url); } $this->photo_url = ''; } public function getCountryName() { if ($this->country && !empty(self::$countries[$this->country])) { return self::$countries[$this->country]; } else { return ''; } } /** * Get first and last name. */ public static function parseName($name) { $data = []; if (!$name) { return $data; } $name_parts = explode(' ', $name, 2); $data['first_name'] = $name_parts[0]; if (!empty($name_parts[1])) { $data['last_name'] = $name_parts[1]; } return $data; } public static function formatSocialProfile($sp) { if (empty($sp['type']) || !isset(self::$social_type_names[$sp['type']])) { $sp['type'] = self::SOCIAL_TYPE_OTHER; } $sp['type_name'] = self::$social_type_names[$sp['type']]; $sp['value_url'] = $sp['value']; if (!preg_match("/^https?:\/\//i", $sp['value_url'])) { switch ($sp['type']) { case self::SOCIAL_TYPE_TELEGRAM: $sp['value_url'] = 'https://t.me/'.$sp['value']; break; default: $sp['value_url'] = 'http://'.$sp['value_url']; break; } } if (empty($sp['value_url'])) { $sp['value_url'] = ''; } return $sp; } public function setPhotoFromRemoteFile($url) { $headers = get_headers($url); if (!preg_match("/200/", $headers[0])) { return false; } $image_data = \Helper::getRemoteFileContents($url); if (!$image_data) { return false; } $temp_file = \Helper::getTempFileName(); \File::put($temp_file, $image_data); $photo_url = $this->savePhoto($temp_file, \File::mimeType($temp_file)); if ($photo_url) { $this->photo_url = $photo_url; return true; } else { return false; } } public function getChannelName() { return \Eventy::filter('channel.name', '', $this->channel); } /** * Get thread meta value. */ public function getMeta($key, $default = null) { if (isset($this->meta[$key])) { return $this->meta[$key]; } else { return $default; } } /** * Set thread meta value. */ public function setMeta($key, $value) { $meta = $this->meta; $meta[$key] = $value; $this->meta = $meta; } public static function getPhoneTypeName($code) { $phone_types = [ self::PHONE_TYPE_WORK => __('Work'), self::PHONE_TYPE_HOME => __('Home'), self::PHONE_TYPE_OTHER => __('Other'), self::PHONE_TYPE_MOBILE => __('Mobile'), self::PHONE_TYPE_FAX => __('Fax'), self::PHONE_TYPE_PAGER => __('Pager'), ]; return $phone_types[$code] ?? ''; } public static function isDefaultPhoneType($code) { return (self::PHONE_TYPE_WORK == $code); } // Method does not check if the customer // has conversations. public function deleteCustomer() { // Delete emails. Email::where('customer_id', $this->id)->delete(); $this->delete(); } public function getChannels() { if (!$this->channel || !$this->channel_id) { return collect([]); } return CustomerChannel::where('customer_id', $this->id)->get(); } public function addChannel($channel, $channel_id) { // We are doing this to let existing modules not to throw error // and as a flag that this customer has record(s) in cucstomer_channel table. if (!$this->channel || !$this->channel_id) { $this->channel = $channel; $this->channel_id = $channel_id; $this->save(); } return CustomerChannel::create($this->id, $channel, $channel_id); } public static function getCustomerByChannel($channel, $channel_id) { $customer_channel = CustomerChannel::where('channel', $channel) ->where('channel_id', $channel_id) ->first(); if ($customer_channel) { return $customer_channel->customer; } else { return null; } } public function getChannelId($channel) { return CustomerChannel::where('customer_id', $this->id) ->where('channel', $channel) ->value('channel_id'); } public static function findCustomersBySocialProfile($type, $value, $exclude_channel = null) { $value = mb_strtolower($value); $like = '%'.$value.'","type":'.$type.'}]'; $customers = Customer::where('social_profiles', \Helper::sqlLikeOperator(), $like)->get(); // Now more prcise filtering. foreach ($customers as $i => $customer) { $ok = false; foreach ($customer->getSocialProfiles() as $social_profile) { if ($social_profile['type'] == $type // Try to check username written in different ways: // - username // - @username // - https://example.org/username && preg_match("#(^|/|@)".preg_quote($value)."$#", trim(mb_strtolower($social_profile['value']))) ) { $ok = true; break; } } if (!$ok) { $customers->forget($i); } } if ($exclude_channel && count($customers)) { $exclude_customer_ids = CustomerChannel::whereIn('customer_id', $customers->pluck('id')) ->where('channel', $exclude_channel) ->pluck('customer_id'); return $customers->whereNotIn('customer_id', $exclude_customer_ids); } else { return $customers; } } }