587 lines
16 KiB
PHP
587 lines
16 KiB
PHP
<?php
|
|
|
|
class Crunchbutton_User extends Cana_Table {
|
|
|
|
public function tipper() {
|
|
// returns a weighted tipper value. 0 = unknown. 5 = the best
|
|
if (!isset($this->_tipper)) {
|
|
$orders = $this->orders();
|
|
$o = [];
|
|
foreach ($orders as $order) {
|
|
if ($order->delivery_type == 'delivery' && $order->pay_type == 'card' && $order->tip) {
|
|
if ($order->tip_type == 'number') {
|
|
$o[] = round(($order->tip / $order->price) * 100);
|
|
} else if ($order->tip_type == 'percent') {
|
|
$o[] = $order->tip;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!count($o)) {
|
|
$tipper = 0;
|
|
} else {
|
|
$score = array_sum($o) / count($o);
|
|
}
|
|
|
|
if ($score < 5) {
|
|
$tipper = 1;
|
|
} else if ($score < 10) {
|
|
$tipper = 2;
|
|
} else if ($score < 15) {
|
|
$tipper = 3;
|
|
} else if ($score < 20) {
|
|
$tipper = 4;
|
|
} else {
|
|
$tipper = 5;
|
|
}
|
|
$this->_tipper = $tipper;
|
|
}
|
|
return $this->_tipper;
|
|
}
|
|
|
|
public function name() {
|
|
if (!isset($this->_name)) {
|
|
$name = explode(' ',$this->name);
|
|
$this->_name = $name[0];
|
|
}
|
|
return $this->_name;
|
|
}
|
|
|
|
public function byPhone($phone, $limit = true) {
|
|
$phone = Phone::clean( $phone );
|
|
$user = User::q('select * from `user` INNER JOIN phone p using(id_phone) WHERE p.phone = ? order by id_user desc'. ($limit ? ' limit 1' : ''), [$phone]);
|
|
if( $limit ){
|
|
$user = $user->get( 0 );
|
|
}
|
|
return $user;
|
|
}
|
|
|
|
public static function byEmail( $email ){
|
|
$user = User::q( 'SELECT * FROM user WHERE email = ? ORDER BY id_user DESC LIMIT 1', [ $email ] )->get( 0 );
|
|
if( $user->id_user ){
|
|
return $user;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public function lastOrder() {
|
|
$order = Order::q('select * from `order` where id_user=? and id_user is not null order by id_order desc limit 1', [$this->id_user]);
|
|
return $order;
|
|
}
|
|
|
|
public function orders($type = 'full') {
|
|
if (!$this->id_user) {
|
|
return new Order;
|
|
}
|
|
|
|
if (!isset($this->_orders)) {
|
|
if ($type == 'compact') {
|
|
$q = "
|
|
select 1 as `compressed`, o.date, o.id_order, o.uuid, r.id_restaurant, r.name as restaurant_name, r.permalink as restaurant_permalink, r.timezone as timezone, 'compressed' as type, o.preordered_date, o.preordered from `order` o
|
|
inner join restaurant r on r.id_restaurant = o.id_restaurant
|
|
where
|
|
id_user=?
|
|
and id_user is not null
|
|
order by date desc
|
|
";
|
|
} else {
|
|
$q = 'select * from `order` where id_user=?';
|
|
}
|
|
|
|
$this->_orders = Order::q($q, [$this->id_user]);
|
|
}
|
|
return $this->_orders;
|
|
}
|
|
|
|
public function watched() {
|
|
return Project::q('
|
|
SELECT project.* FROM project
|
|
LEFT JOIN user_project on user_project.id_project=project.id_project
|
|
WHERE user_project.id_user=?
|
|
', [$this->id_user]);
|
|
}
|
|
|
|
public function projects() {
|
|
|
|
}
|
|
|
|
public function password($password) {
|
|
|
|
}
|
|
|
|
public static function facebook($id) {
|
|
return self::q('
|
|
select user.* from user
|
|
left join user_auth using(id_user)
|
|
where
|
|
user_auth.auth=?
|
|
and user_auth.`type`=\'facebook\'
|
|
and user.active=true
|
|
and user_auth.active=true
|
|
',[$id])->get(0);
|
|
}
|
|
|
|
public static function facebookCreate($id, $auth = false) {
|
|
$fbuser = self::facebook($id);
|
|
$user = $auth ? null : c::user();
|
|
|
|
if (!$fbuser->id_user) {
|
|
// we dont have a user, and we need to make one
|
|
if (!$user->id_user) {
|
|
$user = new User;
|
|
$user->active = 1;
|
|
}
|
|
$fb = new Crunchbutton_Auth_Facebook;
|
|
$user->name = $fb->fbuser()->name;
|
|
$user->email = $fb->fbuser()->email;
|
|
$user->saving_from = $user->saving_from.'User::facebookCreate - ';
|
|
$user->save();
|
|
|
|
$userAuth = new User_Auth;
|
|
$userAuth->active = 1;
|
|
$userAuth->id_user = $user->id_user;
|
|
$userAuth->type = 'facebook';
|
|
$userAuth->auth = $fb->fbuser()->id;
|
|
$userAuth->save();
|
|
|
|
if ($user->phone) {
|
|
User_Auth::createPhoneAuthFromFacebook($user->id_user, $user->phone);
|
|
}
|
|
|
|
} elseif ((!$auth && $fbuser->id_user != $user->id_user)) {
|
|
// somehow the user is logged into a crunchbutton account that is NOT associated with the logged in facebook account!!
|
|
// pretend that the facebook user isnt logged in. we trust our crunchbutton account more
|
|
// when loggin in we will never get here since the code to chceck for token is before facebook cookie
|
|
$user = false;
|
|
} else {
|
|
// we have a valid facebook authed user
|
|
$user = $fbuser;
|
|
}
|
|
|
|
return $user;
|
|
}
|
|
|
|
public function auths() {
|
|
if (!isset($this->_auths)) {
|
|
$this->_auths = User_Auth::q('select * from user_auth where id_user=? and active=true', [$this->id_user]);
|
|
}
|
|
return $this->_auths;
|
|
}
|
|
|
|
public function email() {
|
|
if (!isset($this->_email)) {
|
|
$this->_email = null;
|
|
|
|
foreach ($this->auths() as $auth) {
|
|
if ($auth->type == 'local' && strpos($auth->email,'@') !== false) {
|
|
$this->_email = $auth->email;
|
|
}
|
|
}
|
|
}
|
|
return $this->_email;
|
|
}
|
|
|
|
public function presets() {
|
|
if (!isset($this->_presets)) {
|
|
$this->_presets = Preset::q('
|
|
select * from preset where id_user=?
|
|
', [$this->id_user]);
|
|
}
|
|
return $this->_presets;
|
|
}
|
|
|
|
public function preset($id_restaurant) {
|
|
foreach ($this->presets() as $preset) {
|
|
if ($preset->id_restaurant == $id_restaurant) {
|
|
return $preset;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public function phone(){
|
|
if( !$this->_phone ) {
|
|
$phone = Phone::o( $this->id_phone );
|
|
$phone = $phone->phone;
|
|
$this->_phone = $phone;
|
|
}
|
|
return $this->_phone;
|
|
}
|
|
|
|
public function exports($params = []) {
|
|
|
|
$out = $this->properties();
|
|
// $out[ 'last_tip_delivery' ] = Order::lastTipByDelivery( $this->id_user, 'delivery' );
|
|
// $out[ 'last_tip_takeout' ] = Order::lastTipByDelivery( $this->id_user, 'takeout' )
|
|
|
|
$_ignore = [];
|
|
|
|
if( isset( $params[ 'ignore' ] ) ){
|
|
foreach( $params[ 'ignore' ] as $key => $val ){
|
|
$_ignore[ $val ] = true;
|
|
}
|
|
}
|
|
|
|
if ($this->id_user) {
|
|
|
|
if( is_null( $_ignore[ 'tip' ] ) ){
|
|
$out[ 'last_tip_type' ] = Order::lastTipType( $this->id_user );
|
|
$out[ 'last_tip' ] = Order::lastTip( $this->id_user );
|
|
$out['tipper'] = $this->tipper();
|
|
}
|
|
|
|
if( is_null( $_ignore[ 'auth' ] ) ){
|
|
$out[ 'facebook' ] = User_Auth::userHasFacebookAuth( $this->id_user );
|
|
$out[ 'has_auth' ] = User_Auth::userHasAuth( $this->id_user );
|
|
}
|
|
|
|
$out[ 'phone' ] = $this->phone();
|
|
$lastOrder = Order::lastDeliveredOrder( $this->id_user );
|
|
if( $lastOrder->id_restaurant ){
|
|
$communities = [];
|
|
foreach ( $lastOrder->restaurant()->community() as $community ) {
|
|
$communities[] = $community->id_community;
|
|
}
|
|
$out[ 'last_order' ] = array( 'address' => $lastOrder->address, 'communities' => $communities );
|
|
} else {
|
|
$out[ 'last_order' ] = false;
|
|
}
|
|
|
|
$lastOrderEver = $this->lastOrder();
|
|
if( $lastOrderEver->preordered ){
|
|
$out[ 'last_order_pre_ordered' ] = true;
|
|
}
|
|
|
|
$lastNote = $this->getLastNote();
|
|
if( $lastNote ){
|
|
$out['last_notes'] = trim( $lastNote );
|
|
}
|
|
|
|
if( is_null( $_ignore[ 'presets' ] ) ){
|
|
foreach ($this->presets() as $preset) {
|
|
$out['presets'][$preset->id_restaurant] = $preset->exports();
|
|
}
|
|
}
|
|
|
|
// Get user payment type
|
|
$payment_type = $this->payment_type();
|
|
if( $payment_type ){
|
|
$out[ 'card_type' ] = $payment_type->card_type;
|
|
if( $payment_type->card_type != Crunchbutton_User_Payment_Type::CARD_TYPE_CAMPUS_CASH ){
|
|
$out[ 'card' ] = $payment_type->card;
|
|
$out[ 'card_type' ] = $payment_type->card_type;
|
|
$out[ 'card_exp_year' ] = $payment_type->card_exp_year;
|
|
$out[ 'card_exp_month' ] = $payment_type->card_exp_month;
|
|
}
|
|
}
|
|
|
|
$out[ 'points' ] = Crunchbutton_Credit::exportPoints($this);
|
|
}
|
|
|
|
$out['ip'] = c::getIp();
|
|
$out['email'] = $this->email ? $this->email : $this->email();
|
|
|
|
if( $out['card'] ){
|
|
$out['card_ending'] = substr( $out['card'], -4, 4 );
|
|
}
|
|
|
|
if (c::env() == 'beta' || c::env() == 'local') {
|
|
$out['debug'] = true;
|
|
}
|
|
|
|
unset($out['saving_from']);
|
|
unset($out['balanced_id']);
|
|
unset($out['stripe_id']);
|
|
|
|
|
|
$out['image'] = $this->image(false);
|
|
|
|
// blocked
|
|
$out['blocked'] = Crunchbutton_Blocked::isUserBlocked( $this->id_user );
|
|
$out['phone_blocked'] = Crunchbutton_Blocked::isPhoneBlocked( $this->id_phone );
|
|
|
|
// return the auth token
|
|
if ($params['auth']) {
|
|
$out['token'] = c::auth()->session()->token;
|
|
}
|
|
|
|
if ($params['fb_new']) {
|
|
$out['fb_new'] = true;
|
|
}
|
|
|
|
return $out;
|
|
}
|
|
|
|
public function payment_type(){
|
|
return Crunchbutton_User_Payment_Type::getUserPaymentType($this->id_user);
|
|
}
|
|
|
|
public function getLastNote(){
|
|
$lastOrderNotes = $this->lastOrder();
|
|
if( $lastOrderNotes->notes ){
|
|
$notes = $lastOrderNotes->notes;
|
|
// filter to remove a gift card code
|
|
$promos = Crunchbutton_Promo::q( "SELECT * FROM promo p WHERE p.id_user = {$this->id_user}" );
|
|
foreach( $promos as $promo ){
|
|
$notes = str_replace( $promo->code , '', $notes );
|
|
}
|
|
return $notes;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public function nameAbbr(){
|
|
$name = explode( ' ', $this->name );
|
|
$lastName = $name[ count( $name ) - 1 ];
|
|
if( trim( $lastName ) != '' ){
|
|
$lastName = ' ' . $lastName[0];
|
|
} else {
|
|
$lastName = '';
|
|
}
|
|
return $this->firstName() . $lastName;
|
|
}
|
|
|
|
public function firstName(){
|
|
$name = explode( ' ', $this->name );
|
|
if( trim( $name[ 0 ] ) != '' ){
|
|
return $name[ 0 ];
|
|
}
|
|
return $this->name;
|
|
}
|
|
|
|
public function creditsExport(){
|
|
$credits = $this->credits();
|
|
$out = array();
|
|
foreach ( $credits as $credit ) {
|
|
$out[ $credit->id_credit ] = $credit->exports();;
|
|
}
|
|
return $out;
|
|
}
|
|
|
|
public function debitsExport(){
|
|
$debits = $this->debits();
|
|
$out = array();
|
|
foreach ( $debits as $debit ) {
|
|
$out[ $debit->id_credit ] = $debit->exports();;
|
|
}
|
|
return $out;
|
|
}
|
|
|
|
public function inviteCode(){
|
|
if( !$this->invite_code_updated ){
|
|
$new_code = $this->inviteCodeNameBased();
|
|
$this->invite_code_updated = 1;
|
|
$this->invite_code = $new_code;
|
|
$this->save();
|
|
}
|
|
return $this->invite_code;
|
|
}
|
|
|
|
public function inviteCodeNameBased(){
|
|
// changed to be the first name and a four digits number
|
|
$name = preg_replace('/[^A-Za-z0-9 ]/', '', $this->name );
|
|
$options = explode( ' ', $name );
|
|
$name = $options[ 0 ];
|
|
if( trim( $name ) != '' ){
|
|
$code = strtoupper( $name );
|
|
$code .= self::_inviteCodePartGenerator( '0-9', 4 );
|
|
if( !Crunchbutton_Referral::isCodeAlreadyInUse( $code ) ){
|
|
return $code;
|
|
}
|
|
return $this->inviteCodeNameBased();
|
|
} else {
|
|
return self::inviteCodeGenerator();
|
|
}
|
|
/*
|
|
old method - Make customer referral code name+number #5321
|
|
$name = preg_replace('/[^A-Za-z0-9 ]/', '', $this->name );
|
|
$options = explode( ' ', $name );
|
|
$phone = Phone::clean( $this->phone );
|
|
$name = '';
|
|
foreach( $options as $option ){
|
|
$name .= $option;
|
|
$code = strtoupper( $name . $phone );
|
|
if( !Crunchbutton_Referral::isCodeAlreadyInUse( $code ) ){
|
|
return $code;
|
|
}
|
|
}
|
|
return self::inviteCodeGenerator();
|
|
*/
|
|
}
|
|
|
|
public static function inviteCodeGenerator(){
|
|
$random_id_length = 9;
|
|
$rnd_id = self::_inviteCodePartGenerator('a-z', 3).self::_inviteCodePartGenerator('0-9', 3).self::_inviteCodePartGenerator('a-z', 3);
|
|
// make sure the code do not exist
|
|
$user = Crunchbutton_User::byInviteCode( $rnd_id );
|
|
$admin = Crunchbutton_Admin::byInviteCode( $rnd_id );
|
|
if( $user->count() > 0 || $admin->count() ){
|
|
return Crunchbutton_User::inviteCodeGenerator();
|
|
} else {
|
|
return strtoupper($rnd_id);
|
|
}
|
|
}
|
|
|
|
private static function _inviteCodePartGenerator($chars = '123456789qwertyupasdfghjklzxcvbnm', $len = 0) {
|
|
if ($chars == 'a-z') {
|
|
$chars = 'qwertyupasdfghjklzxcvbnm';
|
|
} elseif ($chars == '0-9') {
|
|
$chars = '123456789';
|
|
}
|
|
$rnd_id = '';
|
|
for ($i = 0; $i < $len; $i++) {
|
|
$rnd_id .= $chars[rand(0, strlen($chars) - 1)];
|
|
}
|
|
return $rnd_id;
|
|
}
|
|
|
|
public static function byInviteCode( $code ){
|
|
return Crunchbutton_User::q( 'SELECT * FROM `user` WHERE UPPER( invite_code ) = UPPER(?)', [$code]);
|
|
}
|
|
|
|
public function credits(){
|
|
return Crunchbutton_Credit::creditByUser( $this->id_user );
|
|
}
|
|
|
|
public function debits(){
|
|
return Crunchbutton_Credit::debitByUser( $this->id_user );
|
|
}
|
|
|
|
public function image($gravatar = true) {
|
|
if (!isset($this->_image)) {
|
|
$auths = $this->auths();
|
|
foreach ($auths as $auth) {
|
|
if ($auth->type == 'facebook') {
|
|
$image = 'https://graph.facebook.com/'.$auth->auth.'/picture?type=square&height=200&width=200';
|
|
break;
|
|
}
|
|
}
|
|
if (!$image && $gravatar) {
|
|
foreach ($auths as $auth) {
|
|
if ($auth->type == 'local') {
|
|
$image = 'https://www.gravatar.com/avatar/'.md5(strtolower($auth->email)).'?s=480&d=404';
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
$this->_image = $image;
|
|
}
|
|
return $this->_image;
|
|
}
|
|
|
|
public static function uuid($uuid) {
|
|
return self::q('select * from `user` where uuid=?', [$uuid])->get(0);
|
|
}
|
|
|
|
|
|
public function save($new = false){
|
|
if( !$this->id_phone ){
|
|
$phone = Phone::byPhone( $this->phone );
|
|
$this->id_phone = $phone->id_phone;
|
|
}
|
|
parent::save();
|
|
}
|
|
|
|
public function __construct($id = null) {
|
|
parent::__construct();
|
|
$this
|
|
->table('user')
|
|
->idVar('id_user')
|
|
->load($id);
|
|
}
|
|
|
|
public function notifications() {
|
|
if (!$this->id_user) {
|
|
return false;
|
|
}
|
|
|
|
if (!isset($this->_notifications)) {
|
|
$this->_notifications = User_Notification::q('select * from user_notification where id_user=? and active=true', [$this->id_user]);
|
|
}
|
|
|
|
return $this->_notifications;
|
|
}
|
|
|
|
public function notifyCSAboutChanges( $changes ){
|
|
if( count( $changes ) ){
|
|
$now = new DateTime( 'now', new DateTimeZone( c::config()->timezone ) );
|
|
$now->modify( '-2 hours' );
|
|
$date = $now->format( 'Y-m-d H:i:s' );
|
|
|
|
$lastOrder = Order::q( 'SELECT * FROM `order` WHERE id_user = ? AND date >= ? ORDER BY id_order DESC LIMIT 1', [ $this->id_user, $date ] )->get( 0 );
|
|
|
|
if( $lastOrder->id_order ){
|
|
|
|
$message = 'Customer ' . $this->name . ' just changed their';
|
|
|
|
$adress = false;
|
|
|
|
if( $changes[ 'address' ] && $changes[ 'address' ][ 'old' ] && $changes[ 'address' ][ 'old' ] != $this->address ){
|
|
$message .= ' address from "' . $changes[ 'address' ][ 'old' ];
|
|
$message .= '" to "' . $this->address . '"';
|
|
$adress = true;
|
|
}
|
|
|
|
if( $changes[ 'phone' ] && $changes[ 'phone' ][ 'old' ] && $changes[ 'phone' ][ 'old' ] != $this->phone ){
|
|
if( $adress ){
|
|
$message .= ' and their';
|
|
}
|
|
$message .= ' phone from "' . $changes[ 'phone' ][ 'old' ];
|
|
$message .= '" to "' . $this->phone . '"';
|
|
}
|
|
|
|
$message .= '. Check with the customer to see what info should be used for this current order #' . $lastOrder->id_order . ' placed at ' . $lastOrder->date()->format( 'M jS Y g:i:s A' );
|
|
|
|
$status = $lastOrder->status()->last();
|
|
|
|
if( $status ){
|
|
$message .= '. Order status: ' . $status[ 'status' ];
|
|
if( $status[ 'status' ] == 'accepted' || $status[ 'status' ] == 'pickedup' || $status[ 'status' ] == 'transfered' && $status[ 'driver' ] ){
|
|
$message .= ' by driver: ' . $status[ 'driver' ][ 'name' ] . ' / ' . $status[ 'driver' ][ 'phone' ] . '.';
|
|
$message .= ' Please inform the driver of this change.';
|
|
}
|
|
}
|
|
|
|
Crunchbutton_Support::createNewWarning( [ 'id_order' => $lastOrder->id_order, 'body' => $message, 'bubble' => true ] );
|
|
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
public function setPush($id, $os = 'ios') {
|
|
if (!$this->id_user) {
|
|
return false;
|
|
}
|
|
|
|
$os = $os == 'ios' ? Crunchbutton_User_Notification::TYPE_PUSH_IOS : Crunchbutton_User_Notification::TYPE_PUSH_ANDROID;
|
|
|
|
$notifications = User_Notification::q('
|
|
SELECT * FROM user_notification
|
|
WHERE
|
|
id_user=?
|
|
AND `type`=?
|
|
', [$this->id_user, $os]);
|
|
foreach($notifications as $n) {
|
|
if ($n->value == $id) {
|
|
$exists = true;
|
|
if (!$n->active) {
|
|
$n->active = 1;
|
|
$n->save();
|
|
}
|
|
}
|
|
}
|
|
if (!$exists) {
|
|
$n = new User_Notification([
|
|
'active' => 1,
|
|
'value' => $id,
|
|
'type' => $os,
|
|
'id_user' => $this->id_user
|
|
]);
|
|
$n->save();
|
|
}
|
|
}
|
|
|
|
}
|