更新
This commit is contained in:
parent
01c7ff2ddf
commit
5c8954b7ce
@ -78,7 +78,7 @@ class StoreCartDao extends BaseDao
|
||||
$query->field('product_id,stock,price,unique,sku,image,svip_price');
|
||||
},
|
||||
'merchant' => function ($query) {
|
||||
$query->field('mer_id,mer_name,mer_state,mer_avatar,is_trader,type_id,credit_buy')->with(['type_name']);
|
||||
$query->field('mer_id,mer_name,mer_state,mer_avatar,is_trader,type_id,credit_buy')->with(['type_names']);
|
||||
}
|
||||
])->select();
|
||||
|
||||
|
@ -1,160 +0,0 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
|
||||
// +----------------------------------------------------------------------
|
||||
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: CRMEB Team <admin@crmeb.com>
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
|
||||
namespace app\common\dao\store\order;
|
||||
|
||||
|
||||
use app\common\dao\BaseDao;
|
||||
use app\common\model\BaseModel;
|
||||
use app\common\model\store\order\StoreCart;
|
||||
use app\common\model\user\UserAddress;
|
||||
use think\Collection;
|
||||
use think\db\exception\DataNotFoundException;
|
||||
use think\db\exception\DbException;
|
||||
use think\db\exception\ModelNotFoundException;
|
||||
use think\model\Relation;
|
||||
|
||||
class StoreCartDao extends BaseDao
|
||||
{
|
||||
|
||||
protected function getModel(): string
|
||||
{
|
||||
return StoreCart::class;
|
||||
}
|
||||
|
||||
public function test()
|
||||
{
|
||||
return StoreCart::getDB()->with(['product' => function (Relation $query) {
|
||||
$query->where('store_name', '儿童节礼物');
|
||||
}])->select();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $ids
|
||||
* @param $uid
|
||||
* @param int|null $merId
|
||||
* @return array
|
||||
* @author xaboy
|
||||
* @day 2020/6/5
|
||||
*/
|
||||
public function validIntersection(array $ids, $uid, int $merId = null): array
|
||||
{
|
||||
return StoreCart::getDB()->whereIn('cart_id', $ids)
|
||||
->when($merId, function ($query, $merId) {
|
||||
$query->where('mer_id', $merId);
|
||||
})
|
||||
->where('is_del', 0)->where('is_fail', 0)->where('is_pay', 0)->where('uid', $uid)->column('cart_id');
|
||||
}
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/6/1
|
||||
* @param int $uid
|
||||
* @return mixed
|
||||
*/
|
||||
public function getAll(int $uid)
|
||||
{
|
||||
$query = ($this->getModel())::where(['uid' => $uid, 'is_del' => 0, 'is_new' => 0, 'is_pay' => 0])
|
||||
->with([
|
||||
'product' => function ($query) {
|
||||
$query->field('product_id,image,store_name,is_show,status,is_del,unit_name,price,mer_status,is_used,product_type,once_max_count,once_min_count,pay_limit,mer_svip_status,svip_price_type');
|
||||
},
|
||||
'productAttr' => function ($query) {
|
||||
$query->field('product_id,stock,price,unique,sku,image,svip_price');
|
||||
},
|
||||
'merchant' => function ($query) {
|
||||
$query->field('mer_id,mer_name,mer_state,mer_avatar,is_trader,type_id')->with(['type_name']);
|
||||
}
|
||||
])->select();
|
||||
|
||||
return $query;
|
||||
}
|
||||
|
||||
public function cartIbByData(array $ids, int $uid, ?UserAddress $address)
|
||||
{
|
||||
return StoreCart::getDb()->where('uid', $uid)->with([
|
||||
'product' => function (Relation $query) use ($address) {
|
||||
$query->field('product_id,cate_id,image,store_name,is_show,status,is_del,unit_name,price,mer_status,temp_id,give_coupon_ids,is_gift_bag,is_used,product_type,old_product_id,integral_rate,delivery_way,delivery_free,type,extend,pay_limit,once_max_count,once_min_count,mer_svip_status,svip_price_type');
|
||||
if ($address) {
|
||||
$cityIds = array_filter([$address->province_id, $address->city_id, $address->district_id, $address->street_id]);
|
||||
$query->with(['temp' => ['region' => function (Relation $query) use ($cityIds) {
|
||||
$query->where(function ($query) use ($cityIds) {
|
||||
foreach ($cityIds as $v) {
|
||||
$query->whereOr('city_id', 'like', "%/{$v}/%");
|
||||
}
|
||||
$query->whereOr('city_id', '0');
|
||||
})->order('shipping_template_region_id DESC')->withLimit(1);
|
||||
}, 'undelives' => function ($query) use ($cityIds) {
|
||||
foreach ($cityIds as $v) {
|
||||
$query->whereOr('city_id', 'like', "%/{$v}/%");
|
||||
}
|
||||
}, 'free' => function (Relation $query) use ($cityIds) {
|
||||
foreach ($cityIds as $v) {
|
||||
$query->whereOr('city_id', 'like', "%/{$v}/%");
|
||||
}
|
||||
$query->order('shipping_template_free_id DESC')->withLimit(1);
|
||||
}]]);
|
||||
}
|
||||
},
|
||||
'productAttr' => function (Relation $query) {
|
||||
$query->field('image,extension_one,extension_two,product_id,stock,price,unique,sku,volume,weight,ot_price,cost,svip_price')
|
||||
->append(['bc_extension_one', 'bc_extension_two']);
|
||||
},
|
||||
'merchant' => function (Relation $query) use ($uid) {
|
||||
$query->field('mer_id,mer_name,mer_state,mer_avatar,delivery_way,commission_rate,category_id')->with(['coupon' => function ($query) use ($uid) {
|
||||
$query->where('uid', $uid);
|
||||
},
|
||||
'config' => function ($query) {
|
||||
$query->whereIn('config_key', ['mer_integral_status', 'mer_integral_rate', 'mer_store_stock', 'mer_take_status', 'mer_take_name', 'mer_take_phone', 'mer_take_address', 'mer_take_location', 'mer_take_day', 'mer_take_time']);
|
||||
},
|
||||
'merchantCategory'
|
||||
]);
|
||||
}])->whereIn('cart_id', $ids)->order('product_type DESC,cart_id DESC')->select();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $cartIds
|
||||
* @param int $uid
|
||||
* @author Qinii
|
||||
*/
|
||||
public function batchDelete(array $cartIds, int $uid)
|
||||
{
|
||||
return ($this->getModel()::getDB())->where('uid', $uid)->whereIn('cart_id', $cartIds)->delete();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int $uid
|
||||
* @return mixed
|
||||
* @author Qinii
|
||||
*/
|
||||
public function getCartCount(int $uid)
|
||||
{
|
||||
$data = ($this->getModel()::getDB())->where(['uid' => $uid, 'is_del' => 0, 'is_new' => 0, 'is_pay' => 0])->field('SUM(cart_num) as count')->select();
|
||||
$data[0]['count'] = $data[0]['count'] ? $data[0]['count'] : 0;
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $source
|
||||
* @param array|null $ids
|
||||
* @author xaboy
|
||||
* @day 2020/8/31
|
||||
*/
|
||||
public function getSourcePayInfo($source, ?array $ids = null)
|
||||
{
|
||||
return StoreCart::getDB()->alias('A')->where('A.source', $source)->where('A.is_pay', 1)->when($ids, function ($query, $ids) {
|
||||
$query->whereIn('A.source_id', $ids);
|
||||
})->leftJoin('StoreOrderProduct B', 'A.cart_id = B.cart_id')
|
||||
->field('sum(B.product_num) as pay_num,sum(B.product_price) as pay_price,A.source_id')->group('A.source_id')->select();
|
||||
}
|
||||
}
|
@ -246,7 +246,7 @@ class Merchant extends BaseModel
|
||||
return $this->hasOne(MerchantType::class, 'mer_type_id', 'type_id');
|
||||
}
|
||||
|
||||
public function typeName()
|
||||
public function typeNames()
|
||||
{
|
||||
return $this->merchantType()->bind(['type_name']);
|
||||
}
|
||||
|
@ -1,234 +0,0 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
|
||||
// +----------------------------------------------------------------------
|
||||
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: CRMEB Team <admin@crmeb.com>
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
|
||||
namespace app\common\model\system\merchant;
|
||||
|
||||
|
||||
use app\common\dao\store\product\ProductDao;
|
||||
use app\common\model\BaseModel;
|
||||
use app\common\model\store\coupon\StoreCouponProduct;
|
||||
use app\common\model\store\coupon\StoreCouponUser;
|
||||
use app\common\model\store\product\Product;
|
||||
use app\common\model\store\product\Spu;
|
||||
use app\common\model\system\config\SystemConfigValue;
|
||||
use app\common\model\system\financial\Financial;
|
||||
use app\common\model\system\serve\ServeOrder;
|
||||
use app\common\repositories\store\StoreActivityRepository;
|
||||
|
||||
class Merchant extends BaseModel
|
||||
{
|
||||
|
||||
/**
|
||||
* @return string
|
||||
* @author xaboy
|
||||
* @day 2020-03-30
|
||||
*/
|
||||
public static function tablePk(): string
|
||||
{
|
||||
return 'mer_id';
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string
|
||||
* @author xaboy
|
||||
* @day 2020-03-30
|
||||
*/
|
||||
public static function tableName(): string
|
||||
{
|
||||
return 'merchant';
|
||||
}
|
||||
|
||||
public function getDeliveryWayAttr($value)
|
||||
{
|
||||
if (!$value) return [];
|
||||
return explode(',',$value);
|
||||
}
|
||||
|
||||
public function product()
|
||||
{
|
||||
return $this->hasMany(Product::class, 'mer_id', 'mer_id');
|
||||
}
|
||||
|
||||
public function config()
|
||||
{
|
||||
return $this->hasMany(SystemConfigValue::class, 'mer_id', 'mer_id');
|
||||
}
|
||||
|
||||
public function showProduct()
|
||||
{
|
||||
return $this->hasMany(Product::class, 'mer_id', 'mer_id')
|
||||
->where((new ProductDao())->productShow())
|
||||
->field('mer_id,product_id,store_name,image,price,is_show,status,is_gift_bag,is_good')
|
||||
->order('is_good DESC,sort DESC');
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO 商户列表下的推荐
|
||||
* @return \think\Collection
|
||||
* @author Qinii
|
||||
* @day 4/20/22
|
||||
*/
|
||||
public function getAllRecommendAttr()
|
||||
{
|
||||
$list = Product::where('mer_id', $this['mer_id'])
|
||||
->where((new ProductDao())->productShow())
|
||||
->field('mer_id,product_id,store_name,image,price,is_show,status,is_gift_bag,is_good,cate_id')
|
||||
->order('sort DESC, create_time DESC')
|
||||
->limit(3)
|
||||
->select()->append(['show_svip_info']);
|
||||
if ($list) {
|
||||
$data = [];
|
||||
$make = app()->make(StoreActivityRepository::class);
|
||||
foreach ($list as $item) {
|
||||
$spu_id = Spu::where('product_id',$item->product_id)->where('product_type' ,0)->value('spu_id');
|
||||
$act = $make->getActivityBySpu(StoreActivityRepository::ACTIVITY_TYPE_BORDER,$spu_id,$item['cate_id'],$item['mer_id']);
|
||||
$item['border_pic'] = $act['pic'] ?? '';
|
||||
$data[] = $item;
|
||||
}
|
||||
return $data;
|
||||
}
|
||||
return [];
|
||||
}
|
||||
|
||||
public function getCityRecommendAttr()
|
||||
{
|
||||
$list = Product::where('mer_id', $this['mer_id'])
|
||||
->where((new ProductDao())->productShow())
|
||||
->whereLike('delivery_way',"%1%")
|
||||
->field('mer_id,product_id,store_name,image,price,is_show,status,is_gift_bag,is_good,cate_id')
|
||||
->order('sort DESC, create_time DESC')
|
||||
->limit(3)
|
||||
->select();
|
||||
if ($list) {
|
||||
$data = [];
|
||||
$make = app()->make(StoreActivityRepository::class);
|
||||
foreach ($list as $item) {
|
||||
$spu_id = Spu::where('product_id',$item->product_id)->where('product_type' ,0)->value('spu_id');
|
||||
$act = $make->getActivityBySpu(StoreActivityRepository::ACTIVITY_TYPE_BORDER,$spu_id,$item['cate_id'],$item['mer_id']);
|
||||
$item['border_pic'] = $act['pic'] ?? '';
|
||||
$data[] = $item;
|
||||
}
|
||||
return $data;
|
||||
}
|
||||
return [];
|
||||
}
|
||||
|
||||
|
||||
public function recommend()
|
||||
{
|
||||
return $this->hasMany(Product::class, 'mer_id', 'mer_id')
|
||||
->where((new ProductDao())->productShow())
|
||||
->where('is_good', 1)
|
||||
->field('mer_id,product_id,store_name,image,price,is_show,status,is_gift_bag,is_good,sales,create_time')
|
||||
->order('is_good DESC,sort DESC,create_time DESC')
|
||||
->limit(3);
|
||||
}
|
||||
|
||||
|
||||
public function coupon()
|
||||
{
|
||||
$time = date('Y-m-d H:i:s');
|
||||
return $this->hasMany(StoreCouponUser::class, 'mer_id', 'mer_id')->where('start_time', '<', $time)->where('end_time', '>', $time)
|
||||
->where('is_fail', 0)->where('status', 0)->order('coupon_price DESC, coupon_user_id ASC')
|
||||
->with(['product' => function ($query) {
|
||||
$query->field('coupon_id,product_id');
|
||||
}, 'coupon' => function ($query) {
|
||||
$query->field('coupon_id,type');
|
||||
}]);
|
||||
}
|
||||
|
||||
public function getServicesTypeAttr()
|
||||
{
|
||||
return merchantConfig($this->mer_id,'services_type');
|
||||
}
|
||||
|
||||
public function marginOrder()
|
||||
{
|
||||
return $this->hasOne(ServeOrder::class, 'mer_id','mer_id')->where('type', 10)->order('create_time DESC');
|
||||
}
|
||||
|
||||
public function refundMarginOrder()
|
||||
{
|
||||
return $this->hasOne(Financial::class, 'mer_id', 'mer_id')
|
||||
->where('type',1)
|
||||
->where('status', -1)
|
||||
->order('create_time DESC')
|
||||
->limit(1);
|
||||
}
|
||||
|
||||
public function merchantCategory()
|
||||
{
|
||||
return $this->hasOne(MerchantCategory::class, 'merchant_category_id', 'category_id');
|
||||
}
|
||||
|
||||
public function merchantType()
|
||||
{
|
||||
return $this->hasOne(MerchantType::class, 'mer_type_id', 'type_id');
|
||||
}
|
||||
|
||||
public function typeName()
|
||||
{
|
||||
return $this->merchantType()->bind(['type_name']);
|
||||
}
|
||||
|
||||
public function getMerCommissionRateAttr()
|
||||
{
|
||||
return $this->commission_rate > 0 ? $this->commission_rate : bcmul($this->merchantCategory->commission_rate, 100, 4);
|
||||
}
|
||||
|
||||
public function getOpenReceiptAttr()
|
||||
{
|
||||
return merchantConfig($this->mer_id, 'mer_open_receipt');
|
||||
}
|
||||
|
||||
public function admin()
|
||||
{
|
||||
return $this->hasOne(MerchantAdmin::class, 'mer_id', 'mer_id')->where('level', 0);
|
||||
}
|
||||
|
||||
|
||||
public function searchKeywordAttr($query, $value)
|
||||
{
|
||||
$query->whereLike('mer_name|mer_keyword', "%{$value}%");
|
||||
}
|
||||
|
||||
public function getFinancialAlipayAttr($value)
|
||||
{
|
||||
return $value ? json_decode($value) : $value;
|
||||
}
|
||||
|
||||
public function getFinancialWechatAttr($value)
|
||||
{
|
||||
return $value ? json_decode($value) : $value;
|
||||
}
|
||||
|
||||
public function getFinancialBankAttr($value)
|
||||
{
|
||||
return $value ? json_decode($value) : $value;
|
||||
}
|
||||
|
||||
public function getMerCertificateAttr()
|
||||
{
|
||||
return merchantConfig($this->mer_id, 'mer_certificate');
|
||||
}
|
||||
|
||||
public function getIssetCertificateAttr()
|
||||
{
|
||||
return count(merchantConfig($this->mer_id, 'mer_certificate') ?: []) > 0;
|
||||
}
|
||||
|
||||
public function searchMerIdsAttr($query, $value)
|
||||
{
|
||||
$query->whereIn('mer_id',$value);
|
||||
}
|
||||
}
|
@ -1174,7 +1174,7 @@ class ProductRepository extends BaseRepository
|
||||
'attrValue',
|
||||
'oldAttrValue',
|
||||
'merchant' => function ($query) {
|
||||
$query->with(['type_name'])->append(['isset_certificate','services_type']);
|
||||
$query->with(['type_names'])->append(['isset_certificate','services_type']);
|
||||
},
|
||||
'seckillActive' => function ($query) {
|
||||
$query->field('start_day,end_day,start_time,end_time,product_id');
|
||||
|
@ -147,7 +147,7 @@ class SpuRepository extends BaseRepository
|
||||
|
||||
$query->with([
|
||||
'merchant' => function ($query) {
|
||||
$query->field($this->merchantFiled)->with(['type_name']);
|
||||
$query->field($this->merchantFiled)->with(['type_names']);
|
||||
},
|
||||
'issetCoupon',
|
||||
'product.attrValue',
|
||||
@ -209,7 +209,7 @@ class SpuRepository extends BaseRepository
|
||||
$count = $query->fetchSql(false)->count();
|
||||
$query->with([
|
||||
'merchant' => function ($query) {
|
||||
$query->field($this->merchantFiled)->with(['type_name']);
|
||||
$query->field($this->merchantFiled)->with(['type_names']);
|
||||
},
|
||||
'issetCoupon'
|
||||
]);
|
||||
|
@ -1,531 +0,0 @@
|
||||
<?php
|
||||
// +----------------------------------------------------------------------
|
||||
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
|
||||
// +----------------------------------------------------------------------
|
||||
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: CRMEB Team <admin@crmeb.com>
|
||||
// +----------------------------------------------------------------------
|
||||
namespace app\common\repositories\store\product;
|
||||
|
||||
use app\common\repositories\store\coupon\StoreCouponProductRepository;
|
||||
use app\common\repositories\store\coupon\StoreCouponRepository;
|
||||
use app\common\repositories\store\StoreActivityRepository;
|
||||
use app\common\repositories\user\UserRepository;
|
||||
use crmeb\jobs\SendSmsJob;
|
||||
use crmeb\jobs\SyncProductTopJob;
|
||||
use crmeb\services\CopyCommand;
|
||||
use crmeb\services\RedisCacheService;
|
||||
use think\exception\ValidateException;
|
||||
use think\facade\Log;
|
||||
use app\common\repositories\BaseRepository;
|
||||
use app\common\dao\store\product\SpuDao;
|
||||
use app\common\repositories\store\StoreCategoryRepository;
|
||||
use app\common\repositories\store\StoreSeckillActiveRepository;
|
||||
use app\common\repositories\user\UserVisitRepository;
|
||||
use think\facade\Queue;
|
||||
|
||||
class SpuRepository extends BaseRepository
|
||||
{
|
||||
public $dao;
|
||||
public $merchantFiled = 'mer_id,mer_name,mer_avatar,is_trader,mer_info,mer_keyword,type_id';
|
||||
public $productFiled = 'S.product_id,S.store_name,S.image,activity_id,S.keyword,S.price,S.mer_id,spu_id,S.status,store_info,brand_id,cate_id,unit_name,S.star,S.rank,S.sort,sales,S.product_type,rate,reply_count,extension_type,S.sys_labels,S.mer_labels,P.delivery_way,P.delivery_free,P.ot_price,svip_price_type,stock,mer_svip_status';
|
||||
public function __construct(SpuDao $dao)
|
||||
{
|
||||
$this->dao = $dao;
|
||||
}
|
||||
|
||||
public function create(array $param, int $productId, int $activityId, $productType = 0)
|
||||
{
|
||||
$data = $this->setparam($param, $productId, $activityId, $productType);
|
||||
return $this->dao->create($data);
|
||||
}
|
||||
|
||||
public function baseUpdate(array $param, int $productId, int $activityId, $productType = 0)
|
||||
{
|
||||
if ($productType == 1) {
|
||||
$make = app()->make(StoreSeckillActiveRepository::class);
|
||||
$activityId = $make->getSearch(['product_id' => $productId])->value('seckill_active_id');
|
||||
}
|
||||
$where = [
|
||||
'product_id' => $productId,
|
||||
'activity_id' => $activityId,
|
||||
'product_type' => $productType,
|
||||
];
|
||||
$ret = $this->dao->getSearch($where)->find();
|
||||
if (!$ret) {
|
||||
return $this->create($param, $productId, $activityId, $productType);
|
||||
} else {
|
||||
$data = $this->setparam($param, $productId, $activityId, $productType);
|
||||
|
||||
$value = $data['mer_labels'];
|
||||
if (!empty($value)) {
|
||||
if (!is_array($value)) {
|
||||
$data['mer_labels'] = ',' . $value . ',';
|
||||
} else {
|
||||
$data['mer_labels'] = ',' . implode(',', $value) . ',';
|
||||
}
|
||||
}
|
||||
return $this->dao->update($ret->spu_id, $data);
|
||||
}
|
||||
}
|
||||
|
||||
public function setparam(array $param, $productId, $activityId, $productType)
|
||||
{
|
||||
|
||||
$data = [
|
||||
'product_id' => $productId,
|
||||
'product_type' => $productType ?? 0,
|
||||
'activity_id' => $activityId,
|
||||
'store_name' => $param['store_name'],
|
||||
'keyword' => $param['keyword'] ?? '',
|
||||
'image' => $param['image'],
|
||||
'price' => $param['price'],
|
||||
'status' => 0,
|
||||
'rank' => $param['rank'] ?? 0,
|
||||
'temp_id' => $param['temp_id'],
|
||||
'sort' => $param['sort'] ?? 0,
|
||||
'mer_labels' => $param['mer_labels'] ?? '',
|
||||
];
|
||||
if (isset($param['mer_id'])) $data['mer_id'] = $param['mer_id'];
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO 修改排序
|
||||
* @param $productId
|
||||
* @param $activityId
|
||||
* @param $productType
|
||||
* @param $data
|
||||
* @author Qinii
|
||||
* @day 1/19/21
|
||||
*/
|
||||
public function updateSort($productId, $activityId, $productType, $data)
|
||||
{
|
||||
$where = [
|
||||
'product_id' => $productId,
|
||||
'activity_id' => $activityId,
|
||||
'product_type' => $productType,
|
||||
];
|
||||
$ret = $this->dao->getSearch($where)->find();
|
||||
if ($ret) $this->dao->update($ret['spu_id'], $data);
|
||||
}
|
||||
/**
|
||||
* TODO 移动端列表
|
||||
* @param $where
|
||||
* @param $page
|
||||
* @param $limit
|
||||
* @param $userInfo
|
||||
* @return array
|
||||
* @author Qinii
|
||||
* @day 12/18/20
|
||||
*/
|
||||
public function getApiSearch($where, $page, $limit, $userInfo = null)
|
||||
{
|
||||
if (isset($where['keyword']) && !empty($where['keyword'])) {
|
||||
if (preg_match('/^(\/@[1-9]{1}).*\*\//', $where['keyword'])) {
|
||||
$command = app()->make(CopyCommand::class)->getMassage($where['keyword']);
|
||||
if (!$command || in_array($command['type'], [30, 40])) return ['count' => 0, 'list' => []];
|
||||
if ($userInfo && $command['uid']) app()->make(UserRepository::class)->bindSpread($userInfo, $command['uid']);
|
||||
$where['spu_id'] = $command['id'];
|
||||
unset($where['keyword']);
|
||||
} else {
|
||||
app()->make(UserVisitRepository::class)->searchProduct($userInfo ? $userInfo['uid'] : 0, $where['keyword'], (int)($where['mer_id'] ?? 0));
|
||||
}
|
||||
}
|
||||
$where['spu_status'] = 1;
|
||||
$where['mer_status'] = 1;
|
||||
$query = $this->dao->search($where);
|
||||
|
||||
$query->with([
|
||||
'merchant' => function ($query) {
|
||||
$query->field($this->merchantFiled)->with(['type_name']);
|
||||
},
|
||||
'issetCoupon',
|
||||
]);
|
||||
$productMake = app()->make(ProductRepository::class);
|
||||
$count = $query->count();
|
||||
|
||||
$list = $query->page($page, $limit)->setOption('field', [])->field($this->productFiled)->select();
|
||||
$append = ['stop_time','svip_price','show_svip_info','is_svip_price'];
|
||||
if ($productMake->getUserIsPromoter($userInfo))
|
||||
$append[] = 'max_extension';
|
||||
$list->append($append);
|
||||
$list = $this->getBorderList($list);
|
||||
return compact('count', 'list');
|
||||
}
|
||||
|
||||
public function getBorderList($list)
|
||||
{
|
||||
$make = app()->make(StoreActivityRepository::class);
|
||||
foreach ($list as $item) {
|
||||
$act = $make->getActivityBySpu(StoreActivityRepository::ACTIVITY_TYPE_BORDER,$item['spu_id'],$item['cate_id'],$item['mer_id']);
|
||||
$item['border_pic'] = $act['pic'] ?? '';
|
||||
}
|
||||
return $list;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* TODO 修改状态
|
||||
* @param array $data
|
||||
* @author Qinii
|
||||
* @day 12/18/20
|
||||
*/
|
||||
public function changeStatus(int $id, int $productType)
|
||||
{
|
||||
$make = app()->make(ProductRepository::class);
|
||||
$where = [];
|
||||
$status = 1;
|
||||
try {
|
||||
switch ($productType) {
|
||||
case 0:
|
||||
$where = [
|
||||
'activity_id' => 0,
|
||||
'product_id' => $id,
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 1:
|
||||
$_make = app()->make(StoreSeckillActiveRepository::class);
|
||||
$res = $_make->getSearch(['product_id' => $id])->find();
|
||||
$endday = strtotime($res['end_day']);
|
||||
if ($res['status'] == -1 || $endday < time()) $status = 0;
|
||||
$where = [
|
||||
'activity_id' => $res['seckill_active_id'],
|
||||
'product_id' => $id,
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 2:
|
||||
$_make = app()->make(ProductPresellRepository::class);
|
||||
$res = $_make->getWhere([$_make->getPk() => $id]);
|
||||
|
||||
$endttime = strtotime($res['end_time']);
|
||||
if ($endttime <= time()) {
|
||||
$status = 0;
|
||||
} else {
|
||||
if (
|
||||
$res['product_status'] !== 1 ||
|
||||
$res['status'] !== 1 ||
|
||||
$res['action_status'] !== 1 ||
|
||||
$res['is_del'] !== 0 ||
|
||||
$res['is_show'] !== 1
|
||||
) {
|
||||
$status = 0;
|
||||
}
|
||||
}
|
||||
$where = [
|
||||
'activity_id' => $id,
|
||||
'product_id' => $res['product_id'],
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 3:
|
||||
$_make = app()->make(ProductAssistRepository::class);
|
||||
$res = $_make->getWhere([$_make->getPk() => $id]);
|
||||
|
||||
$endttime = strtotime($res['end_time']);
|
||||
if ($endttime <= time()) {
|
||||
$status = 0;
|
||||
} else {
|
||||
if (
|
||||
$res['product_status'] !== 1 ||
|
||||
$res['status'] !== 1 ||
|
||||
$res['is_show'] !== 1 ||
|
||||
$res['action_status'] !== 1 ||
|
||||
$res['is_del'] !== 0
|
||||
) {
|
||||
$status = 0;
|
||||
}
|
||||
}
|
||||
|
||||
$where = [
|
||||
'activity_id' => $id,
|
||||
'product_id' => $res['product_id'],
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 4:
|
||||
$_make = app()->make(ProductGroupRepository::class);
|
||||
$wher = $_make->actionShow();
|
||||
$wher[$_make->getPk()] = $id;
|
||||
|
||||
$res = $_make->getWhere([$_make->getPk() => $id]);
|
||||
$endttime = strtotime($res['end_time']);
|
||||
if ($endttime <= time()) {
|
||||
$status = 0;
|
||||
} else {
|
||||
if (
|
||||
$res['product_status'] !== 1 ||
|
||||
$res['status'] !== 1 ||
|
||||
$res['is_show'] !== 1 ||
|
||||
$res['action_status'] !== 1 ||
|
||||
$res['is_del'] !== 0
|
||||
) {
|
||||
$status = 0;
|
||||
}
|
||||
}
|
||||
|
||||
$where = [
|
||||
'activity_id' => $id,
|
||||
'product_id' => $res['product_id'],
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
$ret = $make->getWhere(['product_id' => $where['product_id']]);
|
||||
if (!$ret || $ret['status'] !== 1 || $ret['mer_status'] !== 1 || $ret['is_del']) $status = 0;
|
||||
if (in_array($productType, [0, 1]) && ($ret['is_show'] !== 1 || $ret['is_used'] !== 1)) $status = 0;
|
||||
$result = $this->dao->getSearch($where)->find();
|
||||
if (!$result && $ret) $result = $this->create($ret->toArray(), $where['product_id'], $where['activity_id'], $productType);
|
||||
if ($result) $this->dao->update($result['spu_id'], ['status' => $status]);
|
||||
if ($status == 1 && $productType == 0) {
|
||||
Queue(SendSmsJob::class, ['tempId' => 'PRODUCT_INCREASE', 'id' => $id]);
|
||||
}
|
||||
if ($productType == 0) Queue::push(SyncProductTopJob::class,[]);
|
||||
} catch (\Exception $exception) {
|
||||
Log::info($exception->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO 平台编辑商品同步修改
|
||||
* @param int $id
|
||||
* @param int $productId
|
||||
* @param int $productType
|
||||
* @param array $data
|
||||
* @author Qinii
|
||||
* @day 12/18/20
|
||||
*/
|
||||
public function changRank(int $id, int $productId, int $productType, array $data)
|
||||
{
|
||||
$where = [
|
||||
'product_id' => $productId,
|
||||
'product_type' => $productType,
|
||||
'activity_id' => $id,
|
||||
];
|
||||
$res = $this->dao->getWhere($where);
|
||||
if (!$res && $id) $this->changeStatus($id, $productType);
|
||||
$res = $this->dao->getWhere($where);
|
||||
if ($res) {
|
||||
$res->store_name = $data['store_name'];
|
||||
$res->rank = $data['rank'];
|
||||
$res->star = $data['star'] ?? 1;
|
||||
$res->save();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO 同步各类商品到spu表
|
||||
* @param array|null $productType
|
||||
* @author Qinii
|
||||
* @day 12/25/20
|
||||
*/
|
||||
public function updateSpu(?array $productType)
|
||||
{
|
||||
if (!$productType) $productType = [0, 1, 2, 3, 4];
|
||||
$_product_make = app()->make(ProductRepository::class);
|
||||
$data = [];
|
||||
foreach ($productType as $value) {
|
||||
$ret = $_product_make->activitSearch($value);
|
||||
$data = array_merge($data, $ret);
|
||||
}
|
||||
$this->dao->findOrCreateAll($data);
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO 获取活动商品的一级分类
|
||||
* @param $type
|
||||
* @return mixed
|
||||
* @author Qinii +0
|
||||
* @day 1/12/21
|
||||
*/
|
||||
public function getActiveCategory($type)
|
||||
{
|
||||
$pathArr = $this->dao->getActivecategory($type);
|
||||
$path = [];
|
||||
foreach ($pathArr as $item) {
|
||||
$path[] = explode('/', $item)[1];
|
||||
}
|
||||
$path = array_unique($path);
|
||||
$cat = app()->make(StoreCategoryRepository::class)->getSearch(['ids' => $path])->field('store_category_id,cate_name')->select();
|
||||
return $cat;
|
||||
}
|
||||
|
||||
public function getSpuData($id, $productType, $merId)
|
||||
{
|
||||
try {
|
||||
switch ($productType) {
|
||||
case 0:
|
||||
$where = [
|
||||
'activity_id' => 0,
|
||||
'product_id' => $id,
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 1:
|
||||
$_make = app()->make(StoreSeckillActiveRepository::class);
|
||||
$res = $_make->getSearch(['product_id' => $id])->find();
|
||||
$where = [
|
||||
'activity_id' => $res['seckill_active_id'],
|
||||
'product_id' => $id,
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 2:
|
||||
$_make = app()->make(ProductPresellRepository::class);
|
||||
$res = $_make->getWhere([$_make->getPk() => $id]);
|
||||
$where = [
|
||||
'activity_id' => $id,
|
||||
'product_id' => $res['product_id'],
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 3:
|
||||
$_make = app()->make(ProductAssistRepository::class);
|
||||
$res = $_make->getWhere([$_make->getPk() => $id]);
|
||||
$where = [
|
||||
'activity_id' => $id,
|
||||
'product_id' => $res['product_id'],
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
case 4:
|
||||
$_make = app()->make(ProductGroupRepository::class);
|
||||
$where[$_make->getPk()] = $id;
|
||||
$res = $_make->getWhere([$_make->getPk() => $id]);
|
||||
$where = [
|
||||
'activity_id' => $id,
|
||||
'product_id' => $res['product_id'],
|
||||
'product_type' => $productType,
|
||||
];
|
||||
break;
|
||||
default:
|
||||
$where = [
|
||||
'activity_id' => 0,
|
||||
'product_id' => $id,
|
||||
'product_type' => 0,
|
||||
];
|
||||
break;
|
||||
}
|
||||
} catch (\Exception $e) {
|
||||
throw new ValidateException('数据不存在');
|
||||
}
|
||||
if ($merId) $where['mer_id'] = $merId;
|
||||
$result = $this->dao->getSearch($where)->find();
|
||||
if (!$result) throw new ValidateException('数据不存在');
|
||||
return $result;
|
||||
}
|
||||
|
||||
public function setLabels($id, $productType, $data, $merId = 0)
|
||||
{
|
||||
$field = isset($data['sys_labels']) ? 'sys_labels' : 'mer_labels';
|
||||
if ($data[$field]) app()->make(ProductLabelRepository::class)->checkHas($merId, $data[$field]);
|
||||
$ret = $this->getSpuData($id, $productType, $merId);
|
||||
$value = $data[$field] ? $data[$field] : '';
|
||||
$ret->$field = $value;
|
||||
$ret->save();
|
||||
}
|
||||
|
||||
public function batchLabels($ids, $data,$merId)
|
||||
{
|
||||
$ids = is_array($ids) ? $ids : explode(',',$ids);
|
||||
foreach ($ids as $id) {
|
||||
$this->setLabels($id,0,$data,$merId);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public function getApiSearchByCoupon($where, $page, $limit, $userInfo)
|
||||
{
|
||||
$coupon = app()->make(StoreCouponRepository::class)->search(null, [
|
||||
'status' => 1,
|
||||
'coupon_id' => $where['coupon_id']
|
||||
])->find();
|
||||
$data['coupon'] = $coupon;
|
||||
if ($coupon) {
|
||||
switch ($coupon['type']) {
|
||||
case 0:
|
||||
$where['mer_id'] = $coupon['mer_id'];
|
||||
break;
|
||||
case 1:
|
||||
$where['product_ids'] = app()->make(StoreCouponProductRepository::class)->search([
|
||||
'coupon_id' => $where['coupon_id']
|
||||
])->column('product_id');
|
||||
break;
|
||||
case 11:
|
||||
$ids = app()->make(StoreCouponProductRepository::class)->search([
|
||||
'coupon_id' => $where['coupon_id']
|
||||
])->column('product_id');
|
||||
$where['cate_pid'] = $ids;
|
||||
break;
|
||||
case 10:
|
||||
break;
|
||||
case 12:
|
||||
$ids = app()->make(StoreCouponProductRepository::class)->search([
|
||||
'coupon_id' => $where['coupon_id']
|
||||
])->column('product_id');
|
||||
$where['mer_ids'] = $ids;
|
||||
break;
|
||||
}
|
||||
$where['is_coupon'] = 1;
|
||||
$where['order'] = 'star';
|
||||
$where['common'] = 1;
|
||||
$where['svip'] = ($coupon['send_type'] == StoreCouponRepository::GET_COUPON_TYPE_SVIP) ? 1 : '';
|
||||
$product = $this->getApiSearch($where, $page, $limit, $userInfo);
|
||||
}
|
||||
|
||||
$data['count'] = $product['count'] ?? 0;
|
||||
$data['list'] = $product['list'] ?? [];
|
||||
return $data;
|
||||
}
|
||||
|
||||
public function getHotRanking(int $cateId)
|
||||
{
|
||||
$RedisCacheService = app()->make(RedisCacheService::class);
|
||||
$prefix = env('queue_name','merchant').'_hot_ranking_';
|
||||
$ids = $RedisCacheService->handler()->get($prefix.'top_' . intval($cateId));
|
||||
$ids = $ids ? explode(',', $ids) : [];
|
||||
if (!count($ids)) {
|
||||
return [];
|
||||
}
|
||||
$ids = array_map('intval', $ids);
|
||||
$where['mer_status'] = 1;
|
||||
$where['status'] = 1;
|
||||
$where['is_del'] = 0;
|
||||
$where['product_type'] = 0;
|
||||
$where['order'] = 'sales';
|
||||
$where['spu_ids'] = $ids;
|
||||
$list = $this->dao->search($where)->setOption('field',[])->field('spu_id,S.image,S.price,S.product_type,P.product_id,P.sales,S.status,S.store_name,P.ot_price,P.cost')->select();
|
||||
if ($list) $list = $list->toArray();
|
||||
return $list;
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO
|
||||
* @param $where
|
||||
* @param $page
|
||||
* @param $limit
|
||||
* @return array
|
||||
* @author Qinii
|
||||
* @day 2022/9/22
|
||||
*/
|
||||
public function makinList($where,$page, $limit)
|
||||
{
|
||||
$where['spu_status'] = 1;
|
||||
$where['mer_status'] = 1;
|
||||
$query = $this->dao->search($where);
|
||||
$query->with([
|
||||
'merchant' ,
|
||||
'issetCoupon',
|
||||
]);
|
||||
$count = $query->count();
|
||||
$list = $query->page($page, $limit)->setOption('field', [])->field($this->productFiled)->select();
|
||||
return compact('count','list');
|
||||
}
|
||||
}
|
@ -312,7 +312,7 @@ class MerchantRepository extends BaseRepository
|
||||
if ($where['keyword'] !== '') {
|
||||
app()->make(UserVisitRepository::class)->searchMerchant($userInfo ? $userInfo['uid'] : 0, $where['keyword']);
|
||||
}
|
||||
$query = $this->dao->search($where)->with(['type_name']);
|
||||
$query = $this->dao->search($where)->with(['type_names']);
|
||||
$count = $query->count();
|
||||
$status = systemConfig('mer_location');
|
||||
/** @var MerchantTakeRepository $repository */
|
||||
@ -369,11 +369,8 @@ class MerchantRepository extends BaseRepository
|
||||
*/
|
||||
public function detail($id, $userInfo)
|
||||
{
|
||||
$merchant = $this->dao->apiGetOne($id)->hidden([
|
||||
"real_name", "mer_phone", "reg_admin_id", "sort", "is_del", "is_audit", "is_best", "mer_state", "bank", "bank_number", "bank_name", 'update_time',
|
||||
'financial_alipay', 'financial_bank', 'financial_wechat', 'financial_type','mer_take_phone'
|
||||
]);
|
||||
$merchant->append(['type_name', 'isset_certificate', 'services_type']);
|
||||
$merchant = $this->dao->apiGetOne($id);
|
||||
$merchant->append(['type_names', 'isset_certificate', 'services_type']);
|
||||
$merchant['care'] = false;
|
||||
if ($userInfo)
|
||||
$merchant['care'] = $this->getCareByUser($id, $userInfo->uid);
|
||||
|
@ -1,621 +0,0 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
|
||||
// +----------------------------------------------------------------------
|
||||
// | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||||
// +----------------------------------------------------------------------
|
||||
// | Author: CRMEB Team <admin@crmeb.com>
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
|
||||
namespace app\common\repositories\system\merchant;
|
||||
|
||||
|
||||
use app\common\dao\system\merchant\MerchantDao;
|
||||
use app\common\model\store\order\StoreOrder;
|
||||
use app\common\model\store\product\ProductReply;
|
||||
use app\common\repositories\BaseRepository;
|
||||
use app\common\repositories\store\coupon\StoreCouponRepository;
|
||||
use app\common\repositories\store\coupon\StoreCouponUserRepository;
|
||||
use app\common\repositories\store\product\ProductCopyRepository;
|
||||
use app\common\repositories\store\product\ProductRepository;
|
||||
use app\common\repositories\store\product\SpuRepository;
|
||||
use app\common\repositories\store\shipping\ShippingTemplateRepository;
|
||||
use app\common\repositories\store\StoreCategoryRepository;
|
||||
use app\common\repositories\system\attachment\AttachmentRepository;
|
||||
use app\common\repositories\user\UserBillRepository;
|
||||
use app\common\repositories\user\UserRelationRepository;
|
||||
use app\common\repositories\user\UserVisitRepository;
|
||||
use app\common\repositories\wechat\RoutineQrcodeRepository;
|
||||
use crmeb\jobs\ClearMerchantStoreJob;
|
||||
use crmeb\services\QrcodeService;
|
||||
use crmeb\services\UploadService;
|
||||
use crmeb\services\WechatService;
|
||||
use FormBuilder\Exception\FormBuilderException;
|
||||
use FormBuilder\Factory\Elm;
|
||||
use FormBuilder\Form;
|
||||
use think\db\exception\DataNotFoundException;
|
||||
use think\db\exception\DbException;
|
||||
use think\db\exception\ModelNotFoundException;
|
||||
use think\Exception;
|
||||
use think\exception\ValidateException;
|
||||
use think\facade\Db;
|
||||
use think\facade\Queue;
|
||||
use think\facade\Route;
|
||||
use think\Model;
|
||||
|
||||
/**
|
||||
* Class MerchantRepository
|
||||
* @package app\common\repositories\system\merchant
|
||||
* @mixin MerchantDao
|
||||
* @author xaboy
|
||||
* @day 2020-04-16
|
||||
*/
|
||||
class MerchantRepository extends BaseRepository
|
||||
{
|
||||
|
||||
/**
|
||||
* MerchantRepository constructor.
|
||||
* @param MerchantDao $dao
|
||||
*/
|
||||
public function __construct(MerchantDao $dao)
|
||||
{
|
||||
$this->dao = $dao;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $where
|
||||
* @param $page
|
||||
* @param $limit
|
||||
* @return array
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException
|
||||
* @author xaboy
|
||||
* @day 2020-04-16
|
||||
*/
|
||||
public function lst(array $where, $page, $limit)
|
||||
{
|
||||
$query = $this->dao->search($where);
|
||||
$count = $query->count($this->dao->getPk());
|
||||
$list = $query->page($page, $limit)->setOption('field', [])
|
||||
->with([
|
||||
'admin' => function ($query) {
|
||||
$query->field('mer_id,account');
|
||||
},
|
||||
'merchantCategory',
|
||||
'merchantType'
|
||||
])
|
||||
->field('sort, mer_id, mer_name, real_name, mer_phone, mer_address, mark, status, create_time,is_best,is_trader,type_id,category_id,copy_product_num,export_dump_num,is_margin,margin,mer_avatar')->select();
|
||||
return compact('count', 'list');
|
||||
}
|
||||
|
||||
public function count($where)
|
||||
{
|
||||
$where['status'] = 1;
|
||||
$valid = $this->dao->search($where)->count();
|
||||
$where['status'] = 0;
|
||||
$invalid = $this->dao->search($where)->count();
|
||||
return compact('valid', 'invalid');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|null $id
|
||||
* @param array $formData
|
||||
* @return Form
|
||||
* @throws FormBuilderException
|
||||
* @author xaboy
|
||||
* @day 2020-04-16
|
||||
*/
|
||||
public function form(?int $id = null, array $formData = [])
|
||||
{
|
||||
$form = Elm::createForm(is_null($id) ? Route::buildUrl('systemMerchantCreate')->build() : Route::buildUrl('systemMerchantUpdate', ['id' => $id])->build());
|
||||
$is_margin = 0;
|
||||
if ($formData && $formData['is_margin'] == 10) $is_margin = 1;
|
||||
/** @var MerchantCategoryRepository $make */
|
||||
$make = app()->make(MerchantCategoryRepository::class);
|
||||
$merchantTypeRepository = app()->make(MerchantTypeRepository::class);
|
||||
$options = $merchantTypeRepository->getOptions();
|
||||
$margin = $merchantTypeRepository->getMargin();
|
||||
|
||||
$config = systemConfig(['broadcast_room_type', 'broadcast_goods_type']);
|
||||
|
||||
$rule = [
|
||||
Elm::input('mer_name', '商户名称')->required(),
|
||||
Elm::select('category_id', '商户分类')->options(function () use ($make) {
|
||||
return $make->allOptions();
|
||||
})->requiredNum(),
|
||||
|
||||
Elm::select('type_id', '店铺类型')->disabled($is_margin)->options($options)->requiredNum()->col(12)->control($margin),
|
||||
|
||||
|
||||
Elm::input('mer_account', '商户账号')->required()->disabled(!is_null($id))->required(!is_null($id)),
|
||||
Elm::password('mer_password', '登录密码')->required()->disabled(!is_null($id))->required(!is_null($id)),
|
||||
Elm::input('real_name', '商户姓名'),
|
||||
Elm::input('mer_phone', '商户手机号')->col(12)->required(),
|
||||
Elm::number('commission_rate', '手续费(%)')->col(12),
|
||||
Elm::input('mer_keyword', '商户关键字')->col(12),
|
||||
Elm::input('mer_address', '商户地址'),
|
||||
Elm::input('sub_mchid', '微信分账商户号'),
|
||||
Elm::textarea('mark', '备注'),
|
||||
Elm::number('sort', '排序', 0)->precision(0)->max(99999),
|
||||
$id ? Elm::hidden('status', 1) : Elm::switches('status', '是否开启', 1)->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
Elm::switches('is_bro_room', '直播间审核', $config['broadcast_room_type'] == 1 ? 0 : 1)->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
Elm::switches('is_audit', '产品审核', 1)->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
Elm::switches('is_bro_goods', '直播间商品审核', $config['broadcast_goods_type'] == 1 ? 0 : 1)->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
Elm::switches('is_best', '是否推荐')->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
Elm::switches('is_trader', '是否自营')->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
];
|
||||
|
||||
$form->setRule($rule);
|
||||
return $form->setTitle(is_null($id) ? '添加商户' : '编辑商户')->formData($formData);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $formData
|
||||
* @return Form
|
||||
* @throws FormBuilderException
|
||||
* @author xaboy
|
||||
* @day 2020/6/25
|
||||
*/
|
||||
public function merchantForm(array $formData = [])
|
||||
{
|
||||
$form = Elm::createForm(Route::buildUrl('merchantUpdate')->build());
|
||||
$rule = [
|
||||
Elm::textarea('mer_info', '店铺简介')->required(),
|
||||
Elm::input('service_phone', '服务电话')->required(),
|
||||
Elm::frameImage('mer_banner', '店铺Banner(710*200px)', '/' . config('admin.merchant_prefix') . '/setting/uploadPicture?field=mer_banner&type=1')->modal(['modal' => false])->width('896px')->height('480px')->props(['footer' => false]),
|
||||
Elm::frameImage('mer_avatar', '店铺头像(120*120px)', '/' . config('admin.merchant_prefix') . '/setting/uploadPicture?field=mer_avatar&type=1')->modal(['modal' => false])->width('896px')->height('480px')->props(['footer' => false]),
|
||||
Elm::switches('mer_state', '是否开启', 1)->activeValue(1)->inactiveValue(0)->inactiveText('关')->activeText('开')->col(12),
|
||||
];
|
||||
$form->setRule($rule);
|
||||
return $form->setTitle('编辑店铺信息')->formData($formData);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $id
|
||||
* @return Form
|
||||
* @throws FormBuilderException
|
||||
* @throws DataNotFoundException
|
||||
* @throws DbException
|
||||
* @throws ModelNotFoundException
|
||||
* @author xaboy
|
||||
* @day 2020-04-16
|
||||
*/
|
||||
public function updateForm($id)
|
||||
{
|
||||
$data = $this->dao->get($id)->toArray();
|
||||
/** @var MerchantAdminRepository $make */
|
||||
$make = app()->make(MerchantAdminRepository::class);
|
||||
$data['mer_account'] = $make->merIdByAccount($id);
|
||||
$data['mer_password'] = '***********';
|
||||
if($data['category_id'] == 0){
|
||||
$data['category_id'] = '';
|
||||
}
|
||||
if($data['type_id'] == 0){
|
||||
$data['type_id'] = '';
|
||||
}
|
||||
return $this->form($id, $data);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param array $data
|
||||
* @author xaboy
|
||||
* @day 2020-04-17
|
||||
*/
|
||||
public function createMerchant(array $data)
|
||||
{
|
||||
if ($this->fieldExists('mer_name', $data['mer_name']))
|
||||
throw new ValidateException('商户名已存在');
|
||||
if ($data['mer_phone'] && isPhone($data['mer_phone']))
|
||||
throw new ValidateException('请输入正确的手机号');
|
||||
$merchantCategoryRepository = app()->make(MerchantCategoryRepository::class);
|
||||
$adminRepository = app()->make(MerchantAdminRepository::class);
|
||||
|
||||
if (!$data['category_id'] || !$merchantCategoryRepository->exists($data['category_id']))
|
||||
throw new ValidateException('商户分类不存在');
|
||||
if ($adminRepository->fieldExists('account', $data['mer_account']))
|
||||
throw new ValidateException('账号已存在');
|
||||
|
||||
/** @var MerchantAdminRepository $make */
|
||||
$make = app()->make(MerchantAdminRepository::class);
|
||||
|
||||
$margin = app()->make(MerchantTypeRepository::class)->get($data['type_id']);
|
||||
$data['is_margin'] = $margin['is_margin'] ?? -1;
|
||||
$data['margin'] = $margin['margin'] ?? 0;
|
||||
return Db::transaction(function () use ($data, $make) {
|
||||
$account = $data['mer_account'];
|
||||
$password = $data['mer_password'];
|
||||
unset($data['mer_account'], $data['mer_password']);
|
||||
|
||||
$merchant = $this->dao->create($data);
|
||||
$make->createMerchantAccount($merchant, $account, $password);
|
||||
app()->make(ShippingTemplateRepository::class)->createDefault($merchant->mer_id);
|
||||
app()->make(ProductCopyRepository::class)->defaulCopyNum($merchant->mer_id);
|
||||
return $merchant;
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/5/30
|
||||
* @param $where
|
||||
* @param $page
|
||||
* @param $limit
|
||||
* @return array
|
||||
*/
|
||||
public function getList($where, $page, $limit, $userInfo)
|
||||
{
|
||||
$field = 'care_count,is_trader,type_id,mer_id,mer_banner,mini_banner,mer_name, mark,mer_avatar,product_score,service_score,postage_score,sales,status,is_best,create_time,long,lat,is_margin';
|
||||
$where['status'] = 1;
|
||||
$where['mer_state'] = 1;
|
||||
$where['is_del'] = 0;
|
||||
if (isset($where['location'])) {
|
||||
$data = @explode(',', (string)$where['location']);
|
||||
if (2 != count(array_filter($data ?: []))) {
|
||||
unset($where['location']);
|
||||
} else {
|
||||
$where['location'] = [
|
||||
'lat' => (float)$data[0],
|
||||
'long' => (float)$data[1],
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
if ($where['keyword'] !== '') {
|
||||
app()->make(UserVisitRepository::class)->searchMerchant($userInfo ? $userInfo['uid'] : 0, $where['keyword']);
|
||||
}
|
||||
$query = $this->dao->search($where)->with(['type_name']);
|
||||
$count = $query->count();
|
||||
$status = systemConfig('mer_location');
|
||||
$list = $query->page($page, $limit)->setOption('field', [])->field($field)->select()
|
||||
->each(function ($item) use ($status, $where) {
|
||||
if ($status && $item['lat'] && $item['long'] && isset($where['location']['lat'], $where['location']['long'])) {
|
||||
$distance = getDistance($where['location']['lat'], $where['location']['long'], $item['lat'], $item['long']);
|
||||
if ($distance < 0.9) {
|
||||
$distance = max(bcmul($distance, 1000, 0), 1).'m';
|
||||
if ($distance == '1m') {
|
||||
$distance = '100m以内';
|
||||
}
|
||||
} else {
|
||||
$distance .= 'km';
|
||||
}
|
||||
$item['distance'] = $distance;
|
||||
}
|
||||
$item['recommend'] = isset($where['delivery_way']) ? $item['CityRecommend'] : $item['AllRecommend'];
|
||||
return $item;
|
||||
});
|
||||
|
||||
return compact('count', 'list');
|
||||
}
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/5/30
|
||||
* @param int $id
|
||||
* @return array|Model|null
|
||||
*/
|
||||
public function merExists(int $id)
|
||||
{
|
||||
return ($this->dao->get($id));
|
||||
}
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/5/30
|
||||
* @param $id
|
||||
* @param $userInfo
|
||||
* @return array|Model|null
|
||||
*/
|
||||
public function detail($id, $userInfo)
|
||||
{
|
||||
$merchant = $this->dao->apiGetOne($id)->hidden([
|
||||
"real_name", "mer_phone", "reg_admin_id", "sort", "is_del", "is_audit", "is_best", "mer_state", "bank", "bank_number", "bank_name", 'update_time',
|
||||
'financial_alipay', 'financial_bank', 'financial_wechat', 'financial_type','mer_take_phone'
|
||||
]);
|
||||
$merchant->append(['type_name', 'isset_certificate', 'services_type']);
|
||||
$merchant['care'] = false;
|
||||
if ($userInfo)
|
||||
$merchant['care'] = $this->getCareByUser($id, $userInfo->uid);
|
||||
return $merchant;
|
||||
}
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/5/30
|
||||
* @param int $merId
|
||||
* @param int $userId
|
||||
* @return bool
|
||||
*/
|
||||
public function getCareByUser(int $merId, int $userId)
|
||||
{
|
||||
if (app()->make(UserRelationRepository::class)->getWhere(['type' => 10, 'type_id' => $merId, 'uid' => $userId]))
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/5/30
|
||||
* @param $merId
|
||||
* @param $where
|
||||
* @param $page
|
||||
* @param $limit
|
||||
* @param $userInfo
|
||||
* @return mixed
|
||||
*/
|
||||
public function productList($merId, $where, $page, $limit, $userInfo)
|
||||
{
|
||||
return app()->make(ProductRepository::class)->getApiSearch($merId, $where, $page, $limit, $userInfo);
|
||||
}
|
||||
|
||||
/**
|
||||
* @Author:Qinii
|
||||
* @Date: 2020/5/30
|
||||
* @param int $id
|
||||
* @return mixed
|
||||
*/
|
||||
public function categoryList(int $id)
|
||||
{
|
||||
return app()->make(StoreCategoryRepository::class)->getApiFormatList($id, 1);
|
||||
}
|
||||
|
||||
public function wxQrcode($merId)
|
||||
{
|
||||
$siteUrl = systemConfig('site_url');
|
||||
$name = md5('mwx' . $merId . date('Ymd')) . '.jpg';
|
||||
$attachmentRepository = app()->make(AttachmentRepository::class);
|
||||
$imageInfo = $attachmentRepository->getWhere(['attachment_name' => $name]);
|
||||
|
||||
if (isset($imageInfo['attachment_src']) && strstr($imageInfo['attachment_src'], 'http') !== false && curl_file_exist($imageInfo['attachment_src']) === false) {
|
||||
$imageInfo->delete();
|
||||
$imageInfo = null;
|
||||
}
|
||||
if (!$imageInfo) {
|
||||
$codeUrl = rtrim($siteUrl, '/') . '/pages/store/home/index?id=' . $merId; //二维码链接
|
||||
if (systemConfig('open_wechat_share')) {
|
||||
$qrcode = WechatService::create(false)->qrcodeService();
|
||||
$codeUrl = $qrcode->forever('_scan_url_mer_' . $merId)->url;
|
||||
}
|
||||
$imageInfo = app()->make(QrcodeService::class)->getQRCodePath($codeUrl, $name);
|
||||
if (is_string($imageInfo)) throw new ValidateException('二维码生成失败');
|
||||
|
||||
$imageInfo['dir'] = tidy_url($imageInfo['dir'], null, $siteUrl);
|
||||
|
||||
$attachmentRepository->create(systemConfig('upload_type') ?: 1, -2, $merId, [
|
||||
'attachment_category_id' => 0,
|
||||
'attachment_name' => $imageInfo['name'],
|
||||
'attachment_src' => $imageInfo['dir']
|
||||
]);
|
||||
$urlCode = $imageInfo['dir'];
|
||||
} else $urlCode = $imageInfo['attachment_src'];
|
||||
return $urlCode;
|
||||
}
|
||||
|
||||
public function routineQrcode($merId)
|
||||
{
|
||||
$name = md5('smrt' . $merId . date('Ymd')) . '.jpg';
|
||||
return tidy_url(app()->make(QrcodeService::class)->getRoutineQrcodePath($name, 'pages/store/home/index', 'id=' . $merId), 0);
|
||||
}
|
||||
|
||||
public function copyForm(int $id)
|
||||
{
|
||||
$form = Elm::createForm(Route::buildUrl('systemMerchantChangeCopy', ['id' => $id])->build());
|
||||
$form->setRule([
|
||||
Elm::input('copy_num', '复制次数', $this->dao->getCopyNum($id))->disabled(true)->readonly(true),
|
||||
Elm::radio('type', '修改类型', 1)
|
||||
->setOptions([
|
||||
['value' => 1, 'label' => '增加'],
|
||||
['value' => 2, 'label' => '减少'],
|
||||
]),
|
||||
Elm::number('num', '修改数量', 0)->required()
|
||||
]);
|
||||
return $form->setTitle('修改复制商品次数');
|
||||
}
|
||||
|
||||
public function delete($id)
|
||||
{
|
||||
Db::transaction(function () use ($id) {
|
||||
$this->dao->update($id, ['is_del' => 1]);
|
||||
app()->make(MerchantAdminRepository::class)->deleteMer($id);
|
||||
Queue::push(ClearMerchantStoreJob::class, ['mer_id' => $id]);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO 清理删除商户但没有删除的商品数据
|
||||
* @author Qinii
|
||||
* @day 5/15/21
|
||||
*/
|
||||
public function clearRedundancy()
|
||||
{
|
||||
$ret = (int)$this->dao->search(['is_del' => 1])->value('mer_id');
|
||||
if (!$ret) return;
|
||||
try {
|
||||
app()->make(ProductRepository::class)->clearMerchantProduct($ret);
|
||||
app()->make(StoreCouponRepository::class)->getSearch([])->where('mer_id', $ret)->update(['is_del' => 1, 'status' => 0]);
|
||||
app()->make(StoreCouponUserRepository::class)->getSearch([])->where('mer_id', $ret)->update(['is_fail' => 1, 'status'=>2]);
|
||||
} catch (\Exception $exception) {
|
||||
throw new ValidateException($exception->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
public function addLockMoney(int $merId, string $orderType, int $orderId, float $money)
|
||||
{
|
||||
if ($money <= 0) return;
|
||||
if (systemConfig('mer_lock_time')) {
|
||||
app()->make(UserBillRepository::class)->incBill($merId, 'mer_lock_money', $orderType, [
|
||||
'link_id' => ($orderType === 'order' ? 1 : 2) . $orderId,
|
||||
'mer_id' => $merId,
|
||||
'status' => 0,
|
||||
'title' => '商户冻结余额',
|
||||
'number' => $money,
|
||||
'mark' => '商户冻结余额',
|
||||
'balance' => 0
|
||||
]);
|
||||
} else {
|
||||
$this->dao->addMoney($merId, $money);
|
||||
}
|
||||
}
|
||||
|
||||
public function checkCrmebNum(int $merId, string $type)
|
||||
{
|
||||
$merchant = $this->dao->get($merId);
|
||||
switch ($type) {
|
||||
case 'copy':
|
||||
if (!systemConfig('copy_product_status')) {
|
||||
throw new ValidateException('复制商品未开启');
|
||||
}
|
||||
if (!$merchant['copy_product_num']) {
|
||||
throw new ValidateException('复制商品剩余次数不足');
|
||||
}
|
||||
break;
|
||||
case 'dump':
|
||||
if (!systemConfig('crmeb_serve_dump')) {
|
||||
throw new ValidateException('电子面单未开启');
|
||||
}
|
||||
if (!$merchant['export_dump_num']) {
|
||||
throw new ValidateException('电子面单剩余次数不足');
|
||||
}
|
||||
break;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public function subLockMoney(int $merId, string $orderType, int $orderId, float $money)
|
||||
{
|
||||
if ($money <= 0) return;
|
||||
$make = app()->make(UserBillRepository::class);
|
||||
$bill = $make->search(['category' => 'mer_lock_money', 'type' => $orderType, 'mer_id' => $merId, 'link_id' => ($orderType === 'order' ? 1 : 2) . $orderId, 'status' => 0])->find();
|
||||
if (!$bill) {
|
||||
$this->dao->subMoney($merId, $money);
|
||||
} else {
|
||||
$make->decBill($merId, 'mer_refund_money', $orderType, [
|
||||
'link_id' => ($orderType === 'order' ? 1 : 2) . $orderId,
|
||||
'mer_id' => $merId,
|
||||
'status' => 1,
|
||||
'title' => '商户冻结余额退款',
|
||||
'number' => $money,
|
||||
'mark' => '商户冻结余额退款',
|
||||
'balance' => 0
|
||||
]);
|
||||
}
|
||||
}
|
||||
|
||||
public function computedLockMoney(StoreOrder $order)
|
||||
{
|
||||
Db::transaction(function () use ($order) {
|
||||
$money = 0;
|
||||
$make = app()->make(UserBillRepository::class);
|
||||
$bill = $make->search(['category' => 'mer_lock_money', 'type' => 'order', 'link_id' => '1' . $order->order_id, 'status' => 0])->find();
|
||||
if ($bill) {
|
||||
$money = bcsub($bill->number, $make->refundMerchantMoney($bill->link_id, $bill->type, $bill->mer_id), 2);
|
||||
if ($order->presellOrder) {
|
||||
$presellBill = $make->search(['category' => 'mer_lock_money', 'type' => 'presell', 'link_id' => '2' . $order->presellOrder->presell_order_id, 'status' => 0])->find();
|
||||
if ($presellBill) {
|
||||
$money = bcadd($money, bcsub($presellBill->number, $make->refundMerchantMoney($presellBill->link_id, $presellBill->type, $presellBill->mer_id), 2), 2);
|
||||
$presellBill->status = 1;
|
||||
$presellBill->save();
|
||||
}
|
||||
}
|
||||
$bill->status = 1;
|
||||
$bill->save();
|
||||
}
|
||||
if ($money > 0) {
|
||||
app()->make(UserBillRepository::class)->incBill($order->uid, 'mer_computed_money', 'order', [
|
||||
'link_id' => $order->order_id,
|
||||
'mer_id' => $order->mer_id,
|
||||
'status' => 0,
|
||||
'title' => '商户待解冻余额',
|
||||
'number' => $money,
|
||||
'mark' => '交易完成,商户待解冻余额' . floatval($money) . '元',
|
||||
'balance' => 0
|
||||
]);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public function checkMargin($merId, $typeId)
|
||||
{
|
||||
$merchant = $this->dao->get($merId);
|
||||
$is_margin = 0;
|
||||
$margin = 0;
|
||||
if ($merchant['is_margin'] == 10) {
|
||||
$margin = $merchant['margin'];
|
||||
$is_margin = $merchant['is_margin'];
|
||||
} else {
|
||||
$marginData = app()->make(MerchantTypeRepository::class)->get($typeId);
|
||||
|
||||
if ($marginData) {
|
||||
$is_margin = $marginData['is_margin'];
|
||||
$margin = $marginData['margin'];
|
||||
}
|
||||
}
|
||||
return compact('is_margin', 'margin');
|
||||
}
|
||||
|
||||
public function setMarginForm(int $id)
|
||||
{
|
||||
$merchant = $this->dao->get($id);
|
||||
if ($merchant->is_margin !== 10) {
|
||||
throw new ValidateException('商户无保证金可扣');
|
||||
}
|
||||
$form = Elm::createForm(Route::buildUrl('systemMarginSet')->build());
|
||||
$form->setRule([
|
||||
Elm::input('mer_name', '商户名称', $merchant->mer_name)->disabled(true),
|
||||
Elm::input('mer_id', '商户ID', $merchant->mer_id)->disabled(true),
|
||||
Elm::input('margin', '商户剩余保证金', $merchant->margin)->disabled(true),
|
||||
Elm::number('number', '保证金扣除金额', 0)->max($merchant->margin)->precision(2)->required(),
|
||||
Elm::text('mark', '保证金扣除原因')->required(),
|
||||
]);
|
||||
return $form->setTitle('扣除保证金');
|
||||
}
|
||||
|
||||
/**
|
||||
* TODO
|
||||
* @param $data
|
||||
* @return \think\response\Json
|
||||
* @author Qinii
|
||||
* @day 2/7/22
|
||||
*/
|
||||
public function setMargin($data)
|
||||
{
|
||||
$merechant = $this->dao->get($data['mer_id']);
|
||||
if ($merechant->is_margin !== 10) {
|
||||
throw new ValidateException('商户未支付保证金或已申请退款');
|
||||
}
|
||||
if ($data['number'] < 0) {
|
||||
throw new ValidateException('扣除保证金额不能小于0');
|
||||
}
|
||||
|
||||
if (bccomp($merechant->margin, $data['number'], 2) == -1) {
|
||||
throw new ValidateException('扣除保证金额不足');
|
||||
}
|
||||
|
||||
$data['balance'] = bcsub($merechant->margin, $data['number'], 2);
|
||||
|
||||
Db::transaction(function () use ($merechant, $data) {
|
||||
$merechant->margin = $data['balance'];
|
||||
$merechant->save();
|
||||
app()->make(UserBillRepository::class)->bill(0, 'mer_margin', $data['type'], 0, $data);
|
||||
});
|
||||
}
|
||||
|
||||
public function changeDeliveryBalance($merId,$number)
|
||||
{
|
||||
$merechant = $this->dao->get($merId);
|
||||
if (bccomp($merechant->delivery_balance, $number, 2) == -1) {
|
||||
throw new ValidateException('余额不足,请先充值(配送费用:'.$number.'元)');
|
||||
}
|
||||
Db::transaction(function () use ($merechant, $number) {
|
||||
$merechant->delivery_balance = bcsub($merechant->delivery_balance, $number, 2);
|
||||
$merechant->save();
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
}
|
@ -28,6 +28,7 @@ class paySuccessOrder
|
||||
|
||||
public function handle($event)
|
||||
{
|
||||
Log::info('支付后逻辑22323233');
|
||||
$this->event = $event;
|
||||
$this->finance = [];
|
||||
$this->index = 1;
|
||||
|
@ -57,7 +57,8 @@
|
||||
"overtrue/pinyin": "4.1.0",
|
||||
"jpush/jpush": "^3.6",
|
||||
"guzzlehttp/guzzle": "^6.5",
|
||||
"topthink/think-api": "1.0.27"
|
||||
"topthink/think-api": "1.0.27",
|
||||
"intervention/image": "^2.7"
|
||||
},
|
||||
"require-dev": {
|
||||
"symfony/var-dumper": "^4.2",
|
||||
|
66
composer.lock
generated
66
composer.lock
generated
@ -4,7 +4,7 @@
|
||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
||||
"This file is @generated automatically"
|
||||
],
|
||||
"content-hash": "f33bcb3f1d65204d5920c00b6a27a289",
|
||||
"content-hash": "72a24fa7d140cac6ae5796c4bfe6cffd",
|
||||
"packages": [
|
||||
{
|
||||
"name": "adbario/php-dot-notation",
|
||||
@ -1368,6 +1368,70 @@
|
||||
],
|
||||
"time": "2023-04-17T16:00:37+00:00"
|
||||
},
|
||||
{
|
||||
"name": "intervention/image",
|
||||
"version": "2.7.2",
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://mirrors.cloud.tencent.com/repository/composer/intervention/image/2.7.2/intervention-image-2.7.2.zip",
|
||||
"reference": "04be355f8d6734c826045d02a1079ad658322dad",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"ext-fileinfo": "*",
|
||||
"guzzlehttp/psr7": "~1.1 || ^2.0",
|
||||
"php": ">=5.4.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"mockery/mockery": "~0.9.2",
|
||||
"phpunit/phpunit": "^4.8 || ^5.7 || ^7.5.15"
|
||||
},
|
||||
"suggest": {
|
||||
"ext-gd": "to use GD library based image processing.",
|
||||
"ext-imagick": "to use Imagick based image processing.",
|
||||
"intervention/imagecache": "Caching extension for the Intervention Image library"
|
||||
},
|
||||
"type": "library",
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-master": "2.4-dev"
|
||||
},
|
||||
"laravel": {
|
||||
"providers": [
|
||||
"Intervention\\Image\\ImageServiceProvider"
|
||||
],
|
||||
"aliases": {
|
||||
"Image": "Intervention\\Image\\Facades\\Image"
|
||||
}
|
||||
}
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Intervention\\Image\\": "src/Intervention/Image"
|
||||
}
|
||||
},
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Oliver Vogel",
|
||||
"email": "oliver@intervention.io",
|
||||
"homepage": "https://intervention.io/"
|
||||
}
|
||||
],
|
||||
"description": "Image handling and manipulation library with support for Laravel integration",
|
||||
"homepage": "http://image.intervention.io/",
|
||||
"keywords": [
|
||||
"gd",
|
||||
"image",
|
||||
"imagick",
|
||||
"laravel",
|
||||
"thumbnail",
|
||||
"watermark"
|
||||
],
|
||||
"time": "2022-05-21T17:30:32+00:00"
|
||||
},
|
||||
{
|
||||
"name": "jpush/jpush",
|
||||
"version": "v3.6.8",
|
||||
|
2
vendor/composer/autoload_files.php
vendored
2
vendor/composer/autoload_files.php
vendored
@ -7,8 +7,8 @@ $baseDir = dirname($vendorDir);
|
||||
|
||||
return array(
|
||||
'7b11c4dc42b3b3023073cb14e519683c' => $vendorDir . '/ralouphie/getallheaders/src/getallheaders.php',
|
||||
'e69f7f6ee287b969198c3c9d6777bd38' => $vendorDir . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
||||
'a0edc8309cc5e1d60e3047b5df6b7052' => $vendorDir . '/guzzlehttp/psr7/src/functions_include.php',
|
||||
'e69f7f6ee287b969198c3c9d6777bd38' => $vendorDir . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
||||
'c964ee0ededf28c96ebd9db5099ef910' => $vendorDir . '/guzzlehttp/promises/src/functions_include.php',
|
||||
'25072dd6e2470089de65ae7bf11d3109' => $vendorDir . '/symfony/polyfill-php72/bootstrap.php',
|
||||
'f598d06aa772fa33d905e87be6398fb1' => $vendorDir . '/symfony/polyfill-intl-idn/bootstrap.php',
|
||||
|
1
vendor/composer/autoload_psr4.php
vendored
1
vendor/composer/autoload_psr4.php
vendored
@ -61,6 +61,7 @@ return array(
|
||||
'League\\Flysystem\\Cached\\' => array($vendorDir . '/league/flysystem-cached-adapter/src'),
|
||||
'League\\Flysystem\\' => array($vendorDir . '/league/flysystem/src'),
|
||||
'JPush\\' => array($vendorDir . '/jpush/jpush/src/JPush'),
|
||||
'Intervention\\Image\\' => array($vendorDir . '/intervention/image/src/Intervention/Image'),
|
||||
'GuzzleHttp\\Psr7\\' => array($vendorDir . '/guzzlehttp/psr7/src'),
|
||||
'GuzzleHttp\\Promise\\' => array($vendorDir . '/guzzlehttp/promises/src'),
|
||||
'GuzzleHttp\\Command\\Guzzle\\' => array($vendorDir . '/guzzlehttp/guzzle-services/src'),
|
||||
|
10
vendor/composer/autoload_static.php
vendored
10
vendor/composer/autoload_static.php
vendored
@ -8,8 +8,8 @@ class ComposerStaticInitb1229d2685c190533aa1234015613f09
|
||||
{
|
||||
public static $files = array (
|
||||
'7b11c4dc42b3b3023073cb14e519683c' => __DIR__ . '/..' . '/ralouphie/getallheaders/src/getallheaders.php',
|
||||
'e69f7f6ee287b969198c3c9d6777bd38' => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
||||
'a0edc8309cc5e1d60e3047b5df6b7052' => __DIR__ . '/..' . '/guzzlehttp/psr7/src/functions_include.php',
|
||||
'e69f7f6ee287b969198c3c9d6777bd38' => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
||||
'c964ee0ededf28c96ebd9db5099ef910' => __DIR__ . '/..' . '/guzzlehttp/promises/src/functions_include.php',
|
||||
'25072dd6e2470089de65ae7bf11d3109' => __DIR__ . '/..' . '/symfony/polyfill-php72/bootstrap.php',
|
||||
'f598d06aa772fa33d905e87be6398fb1' => __DIR__ . '/..' . '/symfony/polyfill-intl-idn/bootstrap.php',
|
||||
@ -130,6 +130,10 @@ class ComposerStaticInitb1229d2685c190533aa1234015613f09
|
||||
array (
|
||||
'JPush\\' => 6,
|
||||
),
|
||||
'I' =>
|
||||
array (
|
||||
'Intervention\\Image\\' => 19,
|
||||
),
|
||||
'G' =>
|
||||
array (
|
||||
'GuzzleHttp\\Psr7\\' => 16,
|
||||
@ -406,6 +410,10 @@ class ComposerStaticInitb1229d2685c190533aa1234015613f09
|
||||
array (
|
||||
0 => __DIR__ . '/..' . '/jpush/jpush/src/JPush',
|
||||
),
|
||||
'Intervention\\Image\\' =>
|
||||
array (
|
||||
0 => __DIR__ . '/..' . '/intervention/image/src/Intervention/Image',
|
||||
),
|
||||
'GuzzleHttp\\Psr7\\' =>
|
||||
array (
|
||||
0 => __DIR__ . '/..' . '/guzzlehttp/psr7/src',
|
||||
|
67
vendor/composer/installed.json
vendored
67
vendor/composer/installed.json
vendored
@ -1443,6 +1443,73 @@
|
||||
],
|
||||
"install-path": "../guzzlehttp/psr7"
|
||||
},
|
||||
{
|
||||
"name": "intervention/image",
|
||||
"version": "2.7.2",
|
||||
"version_normalized": "2.7.2.0",
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://mirrors.cloud.tencent.com/repository/composer/intervention/image/2.7.2/intervention-image-2.7.2.zip",
|
||||
"reference": "04be355f8d6734c826045d02a1079ad658322dad",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"ext-fileinfo": "*",
|
||||
"guzzlehttp/psr7": "~1.1 || ^2.0",
|
||||
"php": ">=5.4.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"mockery/mockery": "~0.9.2",
|
||||
"phpunit/phpunit": "^4.8 || ^5.7 || ^7.5.15"
|
||||
},
|
||||
"suggest": {
|
||||
"ext-gd": "to use GD library based image processing.",
|
||||
"ext-imagick": "to use Imagick based image processing.",
|
||||
"intervention/imagecache": "Caching extension for the Intervention Image library"
|
||||
},
|
||||
"time": "2022-05-21T17:30:32+00:00",
|
||||
"type": "library",
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-master": "2.4-dev"
|
||||
},
|
||||
"laravel": {
|
||||
"providers": [
|
||||
"Intervention\\Image\\ImageServiceProvider"
|
||||
],
|
||||
"aliases": {
|
||||
"Image": "Intervention\\Image\\Facades\\Image"
|
||||
}
|
||||
}
|
||||
},
|
||||
"installation-source": "dist",
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Intervention\\Image\\": "src/Intervention/Image"
|
||||
}
|
||||
},
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Oliver Vogel",
|
||||
"email": "oliver@intervention.io",
|
||||
"homepage": "https://intervention.io/"
|
||||
}
|
||||
],
|
||||
"description": "Image handling and manipulation library with support for Laravel integration",
|
||||
"homepage": "http://image.intervention.io/",
|
||||
"keywords": [
|
||||
"gd",
|
||||
"image",
|
||||
"imagick",
|
||||
"laravel",
|
||||
"thumbnail",
|
||||
"watermark"
|
||||
],
|
||||
"install-path": "../intervention/image"
|
||||
},
|
||||
{
|
||||
"name": "jpush/jpush",
|
||||
"version": "v3.6.8",
|
||||
|
13
vendor/composer/installed.php
vendored
13
vendor/composer/installed.php
vendored
@ -3,7 +3,7 @@
|
||||
'name' => 'topthink/think',
|
||||
'pretty_version' => 'dev-master',
|
||||
'version' => 'dev-master',
|
||||
'reference' => '53dcf129fabc042c56f7ba8ccb2338c5779fd41b',
|
||||
'reference' => '01c7ff2ddf4c63db6736300579ff884445b71d85',
|
||||
'type' => 'project',
|
||||
'install_path' => __DIR__ . '/../../',
|
||||
'aliases' => array(),
|
||||
@ -385,6 +385,15 @@
|
||||
'aliases' => array(),
|
||||
'dev_requirement' => false,
|
||||
),
|
||||
'intervention/image' => array(
|
||||
'pretty_version' => '2.7.2',
|
||||
'version' => '2.7.2.0',
|
||||
'reference' => '04be355f8d6734c826045d02a1079ad658322dad',
|
||||
'type' => 'library',
|
||||
'install_path' => __DIR__ . '/../intervention/image',
|
||||
'aliases' => array(),
|
||||
'dev_requirement' => false,
|
||||
),
|
||||
'jpush/jpush' => array(
|
||||
'pretty_version' => 'v3.6.8',
|
||||
'version' => '3.6.8.0',
|
||||
@ -952,7 +961,7 @@
|
||||
'topthink/think' => array(
|
||||
'pretty_version' => 'dev-master',
|
||||
'version' => 'dev-master',
|
||||
'reference' => '53dcf129fabc042c56f7ba8ccb2338c5779fd41b',
|
||||
'reference' => '01c7ff2ddf4c63db6736300579ff884445b71d85',
|
||||
'type' => 'project',
|
||||
'install_path' => __DIR__ . '/../../',
|
||||
'aliases' => array(),
|
||||
|
9
vendor/intervention/image/LICENSE
vendored
Normal file
9
vendor/intervention/image/LICENSE
vendored
Normal file
@ -0,0 +1,9 @@
|
||||
The MIT License (MIT)
|
||||
|
||||
Copyright (c) 2014 Oliver Vogel
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
47
vendor/intervention/image/composer.json
vendored
Normal file
47
vendor/intervention/image/composer.json
vendored
Normal file
@ -0,0 +1,47 @@
|
||||
{
|
||||
"name": "intervention/image",
|
||||
"description": "Image handling and manipulation library with support for Laravel integration",
|
||||
"homepage": "http://image.intervention.io/",
|
||||
"keywords": ["image", "gd", "imagick", "laravel", "watermark", "thumbnail"],
|
||||
"license": "MIT",
|
||||
"authors": [
|
||||
{
|
||||
"name": "Oliver Vogel",
|
||||
"email": "oliver@intervention.io",
|
||||
"homepage": "https://intervention.io/"
|
||||
}
|
||||
],
|
||||
"require": {
|
||||
"php": ">=5.4.0",
|
||||
"ext-fileinfo": "*",
|
||||
"guzzlehttp/psr7": "~1.1 || ^2.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^4.8 || ^5.7 || ^7.5.15",
|
||||
"mockery/mockery": "~0.9.2"
|
||||
},
|
||||
"suggest": {
|
||||
"ext-gd": "to use GD library based image processing.",
|
||||
"ext-imagick": "to use Imagick based image processing.",
|
||||
"intervention/imagecache": "Caching extension for the Intervention Image library"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Intervention\\Image\\": "src/Intervention/Image"
|
||||
}
|
||||
},
|
||||
"extra": {
|
||||
"branch-alias": {
|
||||
"dev-master": "2.4-dev"
|
||||
},
|
||||
"laravel": {
|
||||
"providers": [
|
||||
"Intervention\\Image\\ImageServiceProvider"
|
||||
],
|
||||
"aliases": {
|
||||
"Image": "Intervention\\Image\\Facades\\Image"
|
||||
}
|
||||
}
|
||||
},
|
||||
"minimum-stability": "stable"
|
||||
}
|
11
vendor/intervention/image/provides.json
vendored
Normal file
11
vendor/intervention/image/provides.json
vendored
Normal file
@ -0,0 +1,11 @@
|
||||
{
|
||||
"providers": [
|
||||
"Intervention\\Image\\ImageServiceProvider"
|
||||
],
|
||||
"aliases": [
|
||||
{
|
||||
"alias": "Image",
|
||||
"facade": "Intervention\\Image\\Facades\\Image"
|
||||
}
|
||||
]
|
||||
}
|
229
vendor/intervention/image/src/Intervention/Image/AbstractColor.php
vendored
Normal file
229
vendor/intervention/image/src/Intervention/Image/AbstractColor.php
vendored
Normal file
@ -0,0 +1,229 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Intervention\Image\Exception\NotReadableException;
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
abstract class AbstractColor
|
||||
{
|
||||
/**
|
||||
* Initiates color object from integer
|
||||
*
|
||||
* @param int $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
abstract public function initFromInteger($value);
|
||||
|
||||
/**
|
||||
* Initiates color object from given array
|
||||
*
|
||||
* @param array $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
abstract public function initFromArray($value);
|
||||
|
||||
/**
|
||||
* Initiates color object from given string
|
||||
*
|
||||
* @param string $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
abstract public function initFromString($value);
|
||||
|
||||
/**
|
||||
* Initiates color object from given ImagickPixel object
|
||||
*
|
||||
* @param ImagickPixel $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
abstract public function initFromObject($value);
|
||||
|
||||
/**
|
||||
* Initiates color object from given R, G and B values
|
||||
*
|
||||
* @param int $r
|
||||
* @param int $g
|
||||
* @param int $b
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
abstract public function initFromRgb($r, $g, $b);
|
||||
|
||||
/**
|
||||
* Initiates color object from given R, G, B and A values
|
||||
*
|
||||
* @param int $r
|
||||
* @param int $g
|
||||
* @param int $b
|
||||
* @param float $a
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
abstract public function initFromRgba($r, $g, $b, $a);
|
||||
|
||||
/**
|
||||
* Calculates integer value of current color instance
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
abstract public function getInt();
|
||||
|
||||
/**
|
||||
* Calculates hexadecimal value of current color instance
|
||||
*
|
||||
* @param string $prefix
|
||||
* @return string
|
||||
*/
|
||||
abstract public function getHex($prefix);
|
||||
|
||||
/**
|
||||
* Calculates RGB(A) in array format of current color instance
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
abstract public function getArray();
|
||||
|
||||
/**
|
||||
* Calculates RGBA in string format of current color instance
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract public function getRgba();
|
||||
|
||||
/**
|
||||
* Determines if current color is different from given color
|
||||
*
|
||||
* @param AbstractColor $color
|
||||
* @param int $tolerance
|
||||
* @return boolean
|
||||
*/
|
||||
abstract public function differs(AbstractColor $color, $tolerance = 0);
|
||||
|
||||
/**
|
||||
* Creates new instance
|
||||
*
|
||||
* @param mixed $value
|
||||
*/
|
||||
public function __construct($value = null)
|
||||
{
|
||||
$this->parse($value);
|
||||
}
|
||||
|
||||
/**
|
||||
* Parses given value as color
|
||||
*
|
||||
* @param mixed $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function parse($value)
|
||||
{
|
||||
switch (true) {
|
||||
|
||||
case is_string($value):
|
||||
$this->initFromString($value);
|
||||
break;
|
||||
|
||||
case is_int($value):
|
||||
$this->initFromInteger($value);
|
||||
break;
|
||||
|
||||
case is_array($value):
|
||||
$this->initFromArray($value);
|
||||
break;
|
||||
|
||||
case is_object($value):
|
||||
$this->initFromObject($value);
|
||||
break;
|
||||
|
||||
case is_null($value):
|
||||
$this->initFromArray([255, 255, 255, 0]);
|
||||
break;
|
||||
|
||||
default:
|
||||
throw new NotReadableException(
|
||||
"Color format ({$value}) cannot be read."
|
||||
);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Formats current color instance into given format
|
||||
*
|
||||
* @param string $type
|
||||
* @return mixed
|
||||
*/
|
||||
public function format($type)
|
||||
{
|
||||
switch (strtolower($type)) {
|
||||
|
||||
case 'rgba':
|
||||
return $this->getRgba();
|
||||
|
||||
case 'hex':
|
||||
return $this->getHex('#');
|
||||
|
||||
case 'int':
|
||||
case 'integer':
|
||||
return $this->getInt();
|
||||
|
||||
case 'array':
|
||||
return $this->getArray();
|
||||
|
||||
case 'obj':
|
||||
case 'object':
|
||||
return $this;
|
||||
|
||||
default:
|
||||
throw new NotSupportedException(
|
||||
"Color format ({$type}) is not supported."
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads RGBA values from string into array
|
||||
*
|
||||
* @param string $value
|
||||
* @return array
|
||||
*/
|
||||
protected function rgbaFromString($value)
|
||||
{
|
||||
$result = false;
|
||||
|
||||
// parse color string in hexidecimal format like #cccccc or cccccc or ccc
|
||||
$hexPattern = '/^#?([a-f0-9]{1,2})([a-f0-9]{1,2})([a-f0-9]{1,2})$/i';
|
||||
|
||||
// parse color string in format rgb(140, 140, 140)
|
||||
$rgbPattern = '/^rgb ?\(([0-9]{1,3}), ?([0-9]{1,3}), ?([0-9]{1,3})\)$/i';
|
||||
|
||||
// parse color string in format rgba(255, 0, 0, 0.5)
|
||||
$rgbaPattern = '/^rgba ?\(([0-9]{1,3}), ?([0-9]{1,3}), ?([0-9]{1,3}), ?([0-9.]{1,4})\)$/i';
|
||||
|
||||
if (preg_match($hexPattern, $value, $matches)) {
|
||||
$result = [];
|
||||
$result[0] = strlen($matches[1]) == '1' ? hexdec($matches[1].$matches[1]) : hexdec($matches[1]);
|
||||
$result[1] = strlen($matches[2]) == '1' ? hexdec($matches[2].$matches[2]) : hexdec($matches[2]);
|
||||
$result[2] = strlen($matches[3]) == '1' ? hexdec($matches[3].$matches[3]) : hexdec($matches[3]);
|
||||
$result[3] = 1;
|
||||
} elseif (preg_match($rgbPattern, $value, $matches)) {
|
||||
$result = [];
|
||||
$result[0] = ($matches[1] >= 0 && $matches[1] <= 255) ? intval($matches[1]) : 0;
|
||||
$result[1] = ($matches[2] >= 0 && $matches[2] <= 255) ? intval($matches[2]) : 0;
|
||||
$result[2] = ($matches[3] >= 0 && $matches[3] <= 255) ? intval($matches[3]) : 0;
|
||||
$result[3] = 1;
|
||||
} elseif (preg_match($rgbaPattern, $value, $matches)) {
|
||||
$result = [];
|
||||
$result[0] = ($matches[1] >= 0 && $matches[1] <= 255) ? intval($matches[1]) : 0;
|
||||
$result[1] = ($matches[2] >= 0 && $matches[2] <= 255) ? intval($matches[2]) : 0;
|
||||
$result[2] = ($matches[3] >= 0 && $matches[3] <= 255) ? intval($matches[3]) : 0;
|
||||
$result[3] = ($matches[4] >= 0 && $matches[4] <= 1) ? $matches[4] : 0;
|
||||
} else {
|
||||
throw new NotReadableException(
|
||||
"Unable to read color ({$value})."
|
||||
);
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
}
|
364
vendor/intervention/image/src/Intervention/Image/AbstractDecoder.php
vendored
Normal file
364
vendor/intervention/image/src/Intervention/Image/AbstractDecoder.php
vendored
Normal file
@ -0,0 +1,364 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use GuzzleHttp\Psr7\Stream;
|
||||
use Intervention\Image\Exception\NotReadableException;
|
||||
use Psr\Http\Message\StreamInterface;
|
||||
|
||||
abstract class AbstractDecoder
|
||||
{
|
||||
/**
|
||||
* Initiates new image from path in filesystem
|
||||
*
|
||||
* @param string $path
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
abstract public function initFromPath($path);
|
||||
|
||||
/**
|
||||
* Initiates new image from binary data
|
||||
*
|
||||
* @param string $data
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
abstract public function initFromBinary($data);
|
||||
|
||||
/**
|
||||
* Initiates new image from GD resource
|
||||
*
|
||||
* @param Resource $resource
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
abstract public function initFromGdResource($resource);
|
||||
|
||||
/**
|
||||
* Initiates new image from Imagick object
|
||||
*
|
||||
* @param \Imagick $object
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
abstract public function initFromImagick(\Imagick $object);
|
||||
|
||||
/**
|
||||
* Buffer of input data
|
||||
*
|
||||
* @var mixed
|
||||
*/
|
||||
private $data;
|
||||
|
||||
/**
|
||||
* Creates new Decoder with data
|
||||
*
|
||||
* @param mixed $data
|
||||
*/
|
||||
public function __construct($data = null)
|
||||
{
|
||||
$this->data = $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Init from given URL
|
||||
*
|
||||
* @param string $url
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromUrl($url)
|
||||
{
|
||||
|
||||
$options = [
|
||||
'http' => [
|
||||
'method'=>"GET",
|
||||
'protocol_version'=>1.1, // force use HTTP 1.1 for service mesh environment with envoy
|
||||
'header'=>"Accept-language: en\r\n".
|
||||
"User-Agent: Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/97.0.4692.71 Safari/537.36\r\n"
|
||||
]
|
||||
];
|
||||
|
||||
$context = stream_context_create($options);
|
||||
|
||||
|
||||
if ($data = @file_get_contents($url, false, $context)) {
|
||||
return $this->initFromBinary($data);
|
||||
}
|
||||
|
||||
throw new NotReadableException(
|
||||
"Unable to init from given url (".$url.")."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Init from given stream
|
||||
*
|
||||
* @param StreamInterface|resource $stream
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromStream($stream)
|
||||
{
|
||||
if (!$stream instanceof StreamInterface) {
|
||||
$stream = new Stream($stream);
|
||||
}
|
||||
|
||||
try {
|
||||
$offset = $stream->tell();
|
||||
} catch (\RuntimeException $e) {
|
||||
$offset = 0;
|
||||
}
|
||||
|
||||
$shouldAndCanSeek = $offset !== 0 && $stream->isSeekable();
|
||||
|
||||
if ($shouldAndCanSeek) {
|
||||
$stream->rewind();
|
||||
}
|
||||
|
||||
try {
|
||||
$data = $stream->getContents();
|
||||
} catch (\RuntimeException $e) {
|
||||
$data = null;
|
||||
}
|
||||
|
||||
if ($shouldAndCanSeek) {
|
||||
$stream->seek($offset);
|
||||
}
|
||||
|
||||
if ($data) {
|
||||
return $this->initFromBinary($data);
|
||||
}
|
||||
|
||||
throw new NotReadableException(
|
||||
"Unable to init from given stream"
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is GD resource
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isGdResource()
|
||||
{
|
||||
if (is_resource($this->data)) {
|
||||
return (get_resource_type($this->data) == 'gd');
|
||||
}
|
||||
|
||||
if ($this->data instanceof \GdImage) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is Imagick object
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isImagick()
|
||||
{
|
||||
return is_a($this->data, 'Imagick');
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is Intervention\Image\Image object
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isInterventionImage()
|
||||
{
|
||||
return is_a($this->data, '\Intervention\Image\Image');
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current data is SplFileInfo object
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isSplFileInfo()
|
||||
{
|
||||
return is_a($this->data, 'SplFileInfo');
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current data is Symfony UploadedFile component
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isSymfonyUpload()
|
||||
{
|
||||
return is_a($this->data, 'Symfony\Component\HttpFoundation\File\UploadedFile');
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is file path
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isFilePath()
|
||||
{
|
||||
if (is_string($this->data)) {
|
||||
try {
|
||||
return is_file($this->data);
|
||||
} catch (\Exception $e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is url
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isUrl()
|
||||
{
|
||||
return (bool) filter_var($this->data, FILTER_VALIDATE_URL);
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is a stream resource
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isStream()
|
||||
{
|
||||
if ($this->data instanceof StreamInterface) return true;
|
||||
if (!is_resource($this->data)) return false;
|
||||
if (get_resource_type($this->data) !== 'stream') return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is binary data
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isBinary()
|
||||
{
|
||||
if (is_string($this->data)) {
|
||||
$mime = finfo_buffer(finfo_open(FILEINFO_MIME_TYPE), $this->data);
|
||||
return (substr($mime, 0, 4) != 'text' && $mime != 'application/x-empty');
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is data-url
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isDataUrl()
|
||||
{
|
||||
$data = $this->decodeDataUrl($this->data);
|
||||
|
||||
return is_null($data) ? false : true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current source data is base64 encoded
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isBase64()
|
||||
{
|
||||
if (!is_string($this->data)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return base64_encode(base64_decode($this->data)) === str_replace(["\n", "\r"], '', $this->data);
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates new Image from Intervention\Image\Image
|
||||
*
|
||||
* @param Image $object
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromInterventionImage($object)
|
||||
{
|
||||
return $object;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parses and decodes binary image data from data-url
|
||||
*
|
||||
* @param string $data_url
|
||||
* @return string
|
||||
*/
|
||||
private function decodeDataUrl($data_url)
|
||||
{
|
||||
if (!is_string($data_url)) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$pattern = "/^data:(?:image\/[a-zA-Z\-\.]+)(?:charset=\".+\")?;base64,(?P<data>.+)$/";
|
||||
preg_match($pattern, str_replace(["\n", "\r"], '', $data_url), $matches);
|
||||
|
||||
if (is_array($matches) && array_key_exists('data', $matches)) {
|
||||
return base64_decode($matches['data']);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates new image from mixed data
|
||||
*
|
||||
* @param mixed $data
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function init($data)
|
||||
{
|
||||
$this->data = $data;
|
||||
|
||||
switch (true) {
|
||||
|
||||
case $this->isGdResource():
|
||||
return $this->initFromGdResource($this->data);
|
||||
|
||||
case $this->isImagick():
|
||||
return $this->initFromImagick($this->data);
|
||||
|
||||
case $this->isInterventionImage():
|
||||
return $this->initFromInterventionImage($this->data);
|
||||
|
||||
case $this->isSplFileInfo():
|
||||
return $this->initFromPath($this->data->getRealPath());
|
||||
|
||||
case $this->isBinary():
|
||||
return $this->initFromBinary($this->data);
|
||||
|
||||
case $this->isUrl():
|
||||
return $this->initFromUrl($this->data);
|
||||
|
||||
case $this->isStream():
|
||||
return $this->initFromStream($this->data);
|
||||
|
||||
case $this->isDataUrl():
|
||||
return $this->initFromBinary($this->decodeDataUrl($this->data));
|
||||
|
||||
case $this->isFilePath():
|
||||
return $this->initFromPath($this->data);
|
||||
|
||||
// isBase64 has to be after isFilePath to prevent false positives
|
||||
case $this->isBase64():
|
||||
return $this->initFromBinary(base64_decode($this->data));
|
||||
|
||||
default:
|
||||
throw new NotReadableException("Image source not readable");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Decoder object transforms to string source data
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function __toString()
|
||||
{
|
||||
return (string) $this->data;
|
||||
}
|
||||
}
|
140
vendor/intervention/image/src/Intervention/Image/AbstractDriver.php
vendored
Normal file
140
vendor/intervention/image/src/Intervention/Image/AbstractDriver.php
vendored
Normal file
@ -0,0 +1,140 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
abstract class AbstractDriver
|
||||
{
|
||||
/**
|
||||
* Decoder instance to init images from
|
||||
*
|
||||
* @var \Intervention\Image\AbstractDecoder
|
||||
*/
|
||||
public $decoder;
|
||||
|
||||
/**
|
||||
* Image encoder instance
|
||||
*
|
||||
* @var \Intervention\Image\AbstractEncoder
|
||||
*/
|
||||
public $encoder;
|
||||
|
||||
/**
|
||||
* Creates new image instance
|
||||
*
|
||||
* @param int $width
|
||||
* @param int $height
|
||||
* @param string $background
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
abstract public function newImage($width, $height, $background);
|
||||
|
||||
/**
|
||||
* Reads given string into color object
|
||||
*
|
||||
* @param string $value
|
||||
* @return AbstractColor
|
||||
*/
|
||||
abstract public function parseColor($value);
|
||||
|
||||
/**
|
||||
* Checks if core module installation is available
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
abstract protected function coreAvailable();
|
||||
|
||||
/**
|
||||
* Returns clone of given core
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function cloneCore($core)
|
||||
{
|
||||
return clone $core;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates new image from given input
|
||||
*
|
||||
* @param mixed $data
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function init($data)
|
||||
{
|
||||
return $this->decoder->init($data);
|
||||
}
|
||||
|
||||
/**
|
||||
* Encodes given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param string $format
|
||||
* @param int $quality
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function encode($image, $format, $quality)
|
||||
{
|
||||
return $this->encoder->process($image, $format, $quality);
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes named command on given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param string $name
|
||||
* @param array $arguments
|
||||
* @return \Intervention\Image\Commands\AbstractCommand
|
||||
*/
|
||||
public function executeCommand($image, $name, $arguments)
|
||||
{
|
||||
$commandName = $this->getCommandClassName($name);
|
||||
$command = new $commandName($arguments);
|
||||
$command->execute($image);
|
||||
|
||||
return $command;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns classname of given command name
|
||||
*
|
||||
* @param string $name
|
||||
* @return string
|
||||
*/
|
||||
private function getCommandClassName($name)
|
||||
{
|
||||
if (extension_loaded('mbstring')) {
|
||||
$name = mb_strtoupper(mb_substr($name, 0, 1)) . mb_substr($name, 1);
|
||||
} else {
|
||||
$name = strtoupper(substr($name, 0, 1)) . substr($name, 1);
|
||||
}
|
||||
|
||||
$drivername = $this->getDriverName();
|
||||
$classnameLocal = sprintf('\Intervention\Image\%s\Commands\%sCommand', $drivername, ucfirst($name));
|
||||
$classnameGlobal = sprintf('\Intervention\Image\Commands\%sCommand', ucfirst($name));
|
||||
|
||||
if (class_exists($classnameLocal)) {
|
||||
return $classnameLocal;
|
||||
} elseif (class_exists($classnameGlobal)) {
|
||||
return $classnameGlobal;
|
||||
}
|
||||
|
||||
throw new NotSupportedException(
|
||||
"Command ({$name}) is not available for driver ({$drivername})."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns name of current driver instance
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getDriverName()
|
||||
{
|
||||
$reflect = new \ReflectionClass($this);
|
||||
$namespace = $reflect->getNamespaceName();
|
||||
|
||||
return substr(strrchr($namespace, "\\"), 1);
|
||||
}
|
||||
}
|
271
vendor/intervention/image/src/Intervention/Image/AbstractEncoder.php
vendored
Normal file
271
vendor/intervention/image/src/Intervention/Image/AbstractEncoder.php
vendored
Normal file
@ -0,0 +1,271 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Intervention\Image\Exception\InvalidArgumentException;
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
abstract class AbstractEncoder
|
||||
{
|
||||
/**
|
||||
* Buffer of encode result data
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $result;
|
||||
|
||||
/**
|
||||
* Image object to encode
|
||||
*
|
||||
* @var Image
|
||||
*/
|
||||
public $image;
|
||||
|
||||
/**
|
||||
* Output format of encoder instance
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $format;
|
||||
|
||||
/**
|
||||
* Output quality of encoder instance
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $quality;
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as JPEG string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processJpeg();
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as PNG string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processPng();
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as GIF string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processGif();
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as TIFF string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processTiff();
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as BMP string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processBmp();
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as ICO string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processIco();
|
||||
|
||||
/**
|
||||
* Processes and returns image as WebP encoded string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processWebp();
|
||||
|
||||
/**
|
||||
* Processes and returns image as Avif encoded string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processAvif();
|
||||
|
||||
/**
|
||||
* Processes and returns image as Heic encoded string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
abstract protected function processHeic();
|
||||
|
||||
/**
|
||||
* Process a given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param string $format
|
||||
* @param int $quality
|
||||
* @return Image
|
||||
*/
|
||||
public function process(Image $image, $format = null, $quality = null)
|
||||
{
|
||||
$this->setImage($image);
|
||||
$this->setFormat($format);
|
||||
$this->setQuality($quality);
|
||||
|
||||
switch (strtolower($this->format)) {
|
||||
|
||||
case 'data-url':
|
||||
$this->result = $this->processDataUrl();
|
||||
break;
|
||||
|
||||
case 'gif':
|
||||
case 'image/gif':
|
||||
$this->result = $this->processGif();
|
||||
break;
|
||||
|
||||
case 'png':
|
||||
case 'image/png':
|
||||
case 'image/x-png':
|
||||
$this->result = $this->processPng();
|
||||
break;
|
||||
|
||||
case 'jpg':
|
||||
case 'jpeg':
|
||||
case 'jfif':
|
||||
case 'image/jp2':
|
||||
case 'image/jpg':
|
||||
case 'image/jpeg':
|
||||
case 'image/pjpeg':
|
||||
case 'image/jfif':
|
||||
$this->result = $this->processJpeg();
|
||||
break;
|
||||
|
||||
case 'tif':
|
||||
case 'tiff':
|
||||
case 'image/tiff':
|
||||
case 'image/tif':
|
||||
case 'image/x-tif':
|
||||
case 'image/x-tiff':
|
||||
$this->result = $this->processTiff();
|
||||
break;
|
||||
|
||||
case 'bmp':
|
||||
case 'ms-bmp':
|
||||
case 'x-bitmap':
|
||||
case 'x-bmp':
|
||||
case 'x-ms-bmp':
|
||||
case 'x-win-bitmap':
|
||||
case 'x-windows-bmp':
|
||||
case 'x-xbitmap':
|
||||
case 'image/ms-bmp':
|
||||
case 'image/x-bitmap':
|
||||
case 'image/x-bmp':
|
||||
case 'image/x-ms-bmp':
|
||||
case 'image/x-win-bitmap':
|
||||
case 'image/x-windows-bmp':
|
||||
case 'image/x-xbitmap':
|
||||
$this->result = $this->processBmp();
|
||||
break;
|
||||
|
||||
case 'ico':
|
||||
case 'image/x-ico':
|
||||
case 'image/x-icon':
|
||||
case 'image/vnd.microsoft.icon':
|
||||
$this->result = $this->processIco();
|
||||
break;
|
||||
|
||||
case 'psd':
|
||||
case 'image/vnd.adobe.photoshop':
|
||||
$this->result = $this->processPsd();
|
||||
break;
|
||||
|
||||
case 'webp':
|
||||
case 'image/webp':
|
||||
case 'image/x-webp':
|
||||
$this->result = $this->processWebp();
|
||||
break;
|
||||
|
||||
case 'avif':
|
||||
case 'image/avif':
|
||||
$this->result = $this->processAvif();
|
||||
break;
|
||||
|
||||
case 'heic':
|
||||
case 'image/heic':
|
||||
case 'image/heif':
|
||||
$this->result = $this->processHeic();
|
||||
break;
|
||||
|
||||
default:
|
||||
throw new NotSupportedException(
|
||||
"Encoding format ({$this->format}) is not supported."
|
||||
);
|
||||
}
|
||||
|
||||
$this->setImage(null);
|
||||
|
||||
return $image->setEncoded($this->result);
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as data-url string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processDataUrl()
|
||||
{
|
||||
$mime = $this->image->mime ? $this->image->mime : 'image/png';
|
||||
|
||||
return sprintf('data:%s;base64,%s',
|
||||
$mime,
|
||||
base64_encode($this->process($this->image, $mime, $this->quality))
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets image to process
|
||||
*
|
||||
* @param Image $image
|
||||
*/
|
||||
protected function setImage($image)
|
||||
{
|
||||
$this->image = $image;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines output format
|
||||
*
|
||||
* @param string $format
|
||||
*/
|
||||
protected function setFormat($format = null)
|
||||
{
|
||||
if ($format == '' && $this->image instanceof Image) {
|
||||
$format = $this->image->mime;
|
||||
}
|
||||
|
||||
$this->format = $format ? $format : 'jpg';
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines output quality
|
||||
*
|
||||
* @param int $quality
|
||||
*/
|
||||
protected function setQuality($quality)
|
||||
{
|
||||
$quality = is_null($quality) ? 90 : $quality;
|
||||
$quality = $quality === 0 ? 1 : $quality;
|
||||
|
||||
if ($quality < 0 || $quality > 100) {
|
||||
throw new InvalidArgumentException(
|
||||
'Quality must range from 0 to 100.'
|
||||
);
|
||||
}
|
||||
|
||||
$this->quality = intval($quality);
|
||||
|
||||
return $this;
|
||||
}
|
||||
}
|
295
vendor/intervention/image/src/Intervention/Image/AbstractFont.php
vendored
Normal file
295
vendor/intervention/image/src/Intervention/Image/AbstractFont.php
vendored
Normal file
@ -0,0 +1,295 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
abstract class AbstractFont
|
||||
{
|
||||
/**
|
||||
* Text to be written
|
||||
*
|
||||
* @var String
|
||||
*/
|
||||
public $text;
|
||||
|
||||
/**
|
||||
* Text size in pixels
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $size = 12;
|
||||
|
||||
/**
|
||||
* Color of the text
|
||||
*
|
||||
* @var mixed
|
||||
*/
|
||||
public $color = '000000';
|
||||
|
||||
/**
|
||||
* Rotation angle of the text
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $angle = 0;
|
||||
|
||||
/**
|
||||
* Horizontal alignment of the text
|
||||
*
|
||||
* @var String
|
||||
*/
|
||||
public $align;
|
||||
|
||||
/**
|
||||
* Vertical alignment of the text
|
||||
*
|
||||
* @var String
|
||||
*/
|
||||
public $valign;
|
||||
|
||||
/**
|
||||
* Space between text characters
|
||||
*
|
||||
* @var float
|
||||
*/
|
||||
public $kerning = 0;
|
||||
|
||||
/**
|
||||
* Path to TTF or GD library internal font file of the text
|
||||
*
|
||||
* @var mixed
|
||||
*/
|
||||
public $file;
|
||||
|
||||
/**
|
||||
* Draws font to given image on given position
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $posx
|
||||
* @param int $posy
|
||||
* @return boolean
|
||||
*/
|
||||
abstract public function applyToImage(Image $image, $posx = 0, $posy = 0);
|
||||
|
||||
/**
|
||||
* Calculates bounding box of current font setting
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
abstract public function getBoxSize();
|
||||
|
||||
/**
|
||||
* Create a new instance of Font
|
||||
*
|
||||
* @param String $text Text to be written
|
||||
*/
|
||||
public function __construct($text = null)
|
||||
{
|
||||
$this->text = $text;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set text to be written
|
||||
*
|
||||
* @param String $text
|
||||
* @return self
|
||||
*/
|
||||
public function text($text)
|
||||
{
|
||||
$this->text = $text;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get text to be written
|
||||
*
|
||||
* @return String
|
||||
*/
|
||||
public function getText()
|
||||
{
|
||||
return $this->text;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set font size in pixels
|
||||
*
|
||||
* @param int $size
|
||||
* @return self
|
||||
*/
|
||||
public function size($size)
|
||||
{
|
||||
$this->size = $size;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get font size in pixels
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getSize()
|
||||
{
|
||||
return $this->size;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set color of text to be written
|
||||
*
|
||||
* @param mixed $color
|
||||
* @return self
|
||||
*/
|
||||
public function color($color)
|
||||
{
|
||||
$this->color = $color;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get color of text
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function getColor()
|
||||
{
|
||||
return $this->color;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set rotation angle of text
|
||||
*
|
||||
* @param int $angle
|
||||
* @return self
|
||||
*/
|
||||
public function angle($angle)
|
||||
{
|
||||
$this->angle = $angle;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get rotation angle of text
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getAngle()
|
||||
{
|
||||
return $this->angle;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set horizontal text alignment
|
||||
*
|
||||
* @param string $align
|
||||
* @return self
|
||||
*/
|
||||
public function align($align)
|
||||
{
|
||||
$this->align = $align;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get horizontal text alignment
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getAlign()
|
||||
{
|
||||
return $this->align;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set vertical text alignment
|
||||
*
|
||||
* @param string $valign
|
||||
* @return self
|
||||
*/
|
||||
public function valign($valign)
|
||||
{
|
||||
$this->valign = $valign;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get vertical text alignment
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getValign()
|
||||
{
|
||||
return $this->valign;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set text kerning
|
||||
*
|
||||
* @param string $kerning
|
||||
* @return void
|
||||
*/
|
||||
public function kerning($kerning)
|
||||
{
|
||||
$this->kerning = $kerning;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get kerning
|
||||
*
|
||||
* @return float
|
||||
*/
|
||||
public function getKerning()
|
||||
{
|
||||
return $this->kerning;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set path to font file
|
||||
*
|
||||
* @param string $file
|
||||
* @return self
|
||||
*/
|
||||
public function file($file)
|
||||
{
|
||||
$this->file = $file;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get path to font file
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getFile()
|
||||
{
|
||||
return $this->file;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if current font has access to an applicable font file
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
protected function hasApplicableFontFile()
|
||||
{
|
||||
if (is_string($this->file)) {
|
||||
return file_exists($this->file);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Counts lines of text to be written
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function countLines()
|
||||
{
|
||||
return count(explode(PHP_EOL, $this->text));
|
||||
}
|
||||
}
|
71
vendor/intervention/image/src/Intervention/Image/AbstractShape.php
vendored
Normal file
71
vendor/intervention/image/src/Intervention/Image/AbstractShape.php
vendored
Normal file
@ -0,0 +1,71 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
abstract class AbstractShape
|
||||
{
|
||||
/**
|
||||
* Background color of shape
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $background;
|
||||
|
||||
/**
|
||||
* Border color of current shape
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $border_color;
|
||||
|
||||
/**
|
||||
* Border width of shape
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $border_width = 0;
|
||||
|
||||
/**
|
||||
* Draws shape to given image on given position
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $posx
|
||||
* @param int $posy
|
||||
* @return boolean
|
||||
*/
|
||||
abstract public function applyToImage(Image $image, $posx = 0, $posy = 0);
|
||||
|
||||
/**
|
||||
* Set text to be written
|
||||
*
|
||||
* @param string $text
|
||||
* @return void
|
||||
*/
|
||||
public function background($color)
|
||||
{
|
||||
$this->background = $color;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set border width and color of current shape
|
||||
*
|
||||
* @param int $width
|
||||
* @param string $color
|
||||
* @return void
|
||||
*/
|
||||
public function border($width, $color = null)
|
||||
{
|
||||
$this->border_width = is_numeric($width) ? intval($width) : 0;
|
||||
$this->border_color = is_null($color) ? '#000000' : $color;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current shape has border
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function hasBorder()
|
||||
{
|
||||
return ($this->border_width >= 1);
|
||||
}
|
||||
}
|
81
vendor/intervention/image/src/Intervention/Image/Commands/AbstractCommand.php
vendored
Normal file
81
vendor/intervention/image/src/Intervention/Image/Commands/AbstractCommand.php
vendored
Normal file
@ -0,0 +1,81 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Intervention\Image\Commands\Argument;
|
||||
|
||||
abstract class AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Arguments of command
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
public $arguments;
|
||||
|
||||
/**
|
||||
* Output of command
|
||||
*
|
||||
* @var mixed
|
||||
*/
|
||||
protected $output;
|
||||
|
||||
/**
|
||||
* Executes current command on given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return mixed
|
||||
*/
|
||||
abstract public function execute($image);
|
||||
|
||||
/**
|
||||
* Creates new command instance
|
||||
*
|
||||
* @param array $arguments
|
||||
*/
|
||||
public function __construct($arguments)
|
||||
{
|
||||
$this->arguments = $arguments;
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates new argument instance from given argument key
|
||||
*
|
||||
* @param int $key
|
||||
* @return \Intervention\Image\Commands\Argument
|
||||
*/
|
||||
public function argument($key)
|
||||
{
|
||||
return new Argument($this, $key);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns output data of current command
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function getOutput()
|
||||
{
|
||||
return $this->output ? $this->output : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current instance has output data
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function hasOutput()
|
||||
{
|
||||
return ! is_null($this->output);
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets output data of current command
|
||||
*
|
||||
* @param mixed $value
|
||||
*/
|
||||
public function setOutput($value)
|
||||
{
|
||||
$this->output = $value;
|
||||
}
|
||||
}
|
225
vendor/intervention/image/src/Intervention/Image/Commands/Argument.php
vendored
Normal file
225
vendor/intervention/image/src/Intervention/Image/Commands/Argument.php
vendored
Normal file
@ -0,0 +1,225 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Intervention\Image\Exception\InvalidArgumentException;
|
||||
|
||||
class Argument
|
||||
{
|
||||
/**
|
||||
* Command with arguments
|
||||
*
|
||||
* @var AbstractCommand
|
||||
*/
|
||||
public $command;
|
||||
|
||||
/**
|
||||
* Key of argument in array
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $key;
|
||||
|
||||
/**
|
||||
* Creates new instance from given command and key
|
||||
*
|
||||
* @param AbstractCommand $command
|
||||
* @param int $key
|
||||
*/
|
||||
public function __construct(AbstractCommand $command, $key = 0)
|
||||
{
|
||||
$this->command = $command;
|
||||
$this->key = $key;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns name of current arguments command
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getCommandName()
|
||||
{
|
||||
preg_match("/\\\\([\w]+)Command$/", get_class($this->command), $matches);
|
||||
return isset($matches[1]) ? lcfirst($matches[1]).'()' : 'Method';
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns value of current argument
|
||||
*
|
||||
* @param mixed $default
|
||||
* @return mixed
|
||||
*/
|
||||
public function value($default = null)
|
||||
{
|
||||
$arguments = $this->command->arguments;
|
||||
|
||||
if (is_array($arguments)) {
|
||||
return isset($arguments[$this->key]) ? $arguments[$this->key] : $default;
|
||||
}
|
||||
|
||||
return $default;
|
||||
}
|
||||
|
||||
/**
|
||||
* Defines current argument as required
|
||||
*
|
||||
* @return \Intervention\Image\Commands\Argument
|
||||
*/
|
||||
public function required()
|
||||
{
|
||||
if ( ! array_key_exists($this->key, $this->command->arguments)) {
|
||||
throw new InvalidArgumentException(
|
||||
sprintf("Missing argument %d for %s", $this->key + 1, $this->getCommandName())
|
||||
);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines that current argument must be of given type
|
||||
*
|
||||
* @return \Intervention\Image\Commands\Argument
|
||||
*/
|
||||
public function type($type)
|
||||
{
|
||||
$valid = true;
|
||||
$value = $this->value();
|
||||
|
||||
if ($value === null) {
|
||||
return $this;
|
||||
}
|
||||
|
||||
switch (strtolower($type)) {
|
||||
case 'bool':
|
||||
case 'boolean':
|
||||
$valid = \is_bool($value);
|
||||
$message = '%s accepts only boolean values as argument %d.';
|
||||
break;
|
||||
case 'int':
|
||||
case 'integer':
|
||||
$valid = \is_int($value);
|
||||
$message = '%s accepts only integer values as argument %d.';
|
||||
break;
|
||||
case 'num':
|
||||
case 'numeric':
|
||||
$valid = is_numeric($value);
|
||||
$message = '%s accepts only numeric values as argument %d.';
|
||||
break;
|
||||
case 'str':
|
||||
case 'string':
|
||||
$valid = \is_string($value);
|
||||
$message = '%s accepts only string values as argument %d.';
|
||||
break;
|
||||
case 'array':
|
||||
$valid = \is_array($value);
|
||||
$message = '%s accepts only array as argument %d.';
|
||||
break;
|
||||
case 'closure':
|
||||
$valid = is_a($value, '\Closure');
|
||||
$message = '%s accepts only Closure as argument %d.';
|
||||
break;
|
||||
case 'digit':
|
||||
$valid = $this->isDigit($value);
|
||||
$message = '%s accepts only integer values as argument %d.';
|
||||
break;
|
||||
}
|
||||
|
||||
if (! $valid) {
|
||||
$commandName = $this->getCommandName();
|
||||
$argument = $this->key + 1;
|
||||
|
||||
if (isset($message)) {
|
||||
$message = sprintf($message, $commandName, $argument);
|
||||
} else {
|
||||
$message = sprintf('Missing argument for %d.', $argument);
|
||||
}
|
||||
|
||||
throw new InvalidArgumentException(
|
||||
$message
|
||||
);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines that current argument value must be numeric between given values
|
||||
*
|
||||
* @return \Intervention\Image\Commands\Argument
|
||||
*/
|
||||
public function between($x, $y)
|
||||
{
|
||||
$value = $this->type('numeric')->value();
|
||||
|
||||
if (is_null($value)) {
|
||||
return $this;
|
||||
}
|
||||
|
||||
$alpha = min($x, $y);
|
||||
$omega = max($x, $y);
|
||||
|
||||
if ($value < $alpha || $value > $omega) {
|
||||
throw new InvalidArgumentException(
|
||||
sprintf('Argument %d must be between %s and %s.', $this->key, $x, $y)
|
||||
);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines that current argument must be over a minimum value
|
||||
*
|
||||
* @return \Intervention\Image\Commands\Argument
|
||||
*/
|
||||
public function min($value)
|
||||
{
|
||||
$v = $this->type('numeric')->value();
|
||||
|
||||
if (is_null($v)) {
|
||||
return $this;
|
||||
}
|
||||
|
||||
if ($v < $value) {
|
||||
throw new InvalidArgumentException(
|
||||
sprintf('Argument %d must be at least %s.', $this->key, $value)
|
||||
);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines that current argument must be under a maxiumum value
|
||||
*
|
||||
* @return \Intervention\Image\Commands\Argument
|
||||
*/
|
||||
public function max($value)
|
||||
{
|
||||
$v = $this->type('numeric')->value();
|
||||
|
||||
if (is_null($v)) {
|
||||
return $this;
|
||||
}
|
||||
|
||||
if ($v > $value) {
|
||||
throw new InvalidArgumentException(
|
||||
sprintf('Argument %d may not be greater than %s.', $this->key, $value)
|
||||
);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if value is "PHP" integer (120 but also 120.0)
|
||||
*
|
||||
* @param mixed $value
|
||||
* @return boolean
|
||||
*/
|
||||
private function isDigit($value)
|
||||
{
|
||||
return is_numeric($value) ? intval($value) == $value : false;
|
||||
}
|
||||
}
|
29
vendor/intervention/image/src/Intervention/Image/Commands/ChecksumCommand.php
vendored
Normal file
29
vendor/intervention/image/src/Intervention/Image/Commands/ChecksumCommand.php
vendored
Normal file
@ -0,0 +1,29 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
class ChecksumCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Calculates checksum of given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$colors = [];
|
||||
|
||||
$size = $image->getSize();
|
||||
|
||||
for ($x=0; $x <= ($size->width-1); $x++) {
|
||||
for ($y=0; $y <= ($size->height-1); $y++) {
|
||||
$colors[] = $image->pickColor($x, $y, 'array');
|
||||
}
|
||||
}
|
||||
|
||||
$this->setOutput(md5(serialize($colors)));
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
35
vendor/intervention/image/src/Intervention/Image/Commands/CircleCommand.php
vendored
Normal file
35
vendor/intervention/image/src/Intervention/Image/Commands/CircleCommand.php
vendored
Normal file
@ -0,0 +1,35 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Closure;
|
||||
|
||||
class CircleCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Draw a circle centered on given image
|
||||
*
|
||||
* @param \Intervention\Image\image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$diameter = $this->argument(0)->type('numeric')->required()->value();
|
||||
$x = $this->argument(1)->type('numeric')->required()->value();
|
||||
$y = $this->argument(2)->type('numeric')->required()->value();
|
||||
$callback = $this->argument(3)->type('closure')->value();
|
||||
|
||||
$circle_classname = sprintf('\Intervention\Image\%s\Shapes\CircleShape',
|
||||
$image->getDriver()->getDriverName());
|
||||
|
||||
$circle = new $circle_classname($diameter);
|
||||
|
||||
if ($callback instanceof Closure) {
|
||||
$callback($circle);
|
||||
}
|
||||
|
||||
$circle->applyToImage($image, $x, $y);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
36
vendor/intervention/image/src/Intervention/Image/Commands/EllipseCommand.php
vendored
Normal file
36
vendor/intervention/image/src/Intervention/Image/Commands/EllipseCommand.php
vendored
Normal file
@ -0,0 +1,36 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Closure;
|
||||
|
||||
class EllipseCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Draws ellipse on given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$width = $this->argument(0)->type('numeric')->required()->value();
|
||||
$height = $this->argument(1)->type('numeric')->required()->value();
|
||||
$x = $this->argument(2)->type('numeric')->required()->value();
|
||||
$y = $this->argument(3)->type('numeric')->required()->value();
|
||||
$callback = $this->argument(4)->type('closure')->value();
|
||||
|
||||
$ellipse_classname = sprintf('\Intervention\Image\%s\Shapes\EllipseShape',
|
||||
$image->getDriver()->getDriverName());
|
||||
|
||||
$ellipse = new $ellipse_classname($width, $height);
|
||||
|
||||
if ($callback instanceof Closure) {
|
||||
$callback($ellipse);
|
||||
}
|
||||
|
||||
$ellipse->applyToImage($image, $x, $y);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
61
vendor/intervention/image/src/Intervention/Image/Commands/ExifCommand.php
vendored
Normal file
61
vendor/intervention/image/src/Intervention/Image/Commands/ExifCommand.php
vendored
Normal file
@ -0,0 +1,61 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Intervention\Image\Exception\NotReadableException;
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
class ExifCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Read Exif data from the given image
|
||||
*
|
||||
* Note: Windows PHP Users - in order to use this method you will need to
|
||||
* enable the mbstring and exif extensions within the php.ini file.
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
if (!function_exists('exif_read_data')) {
|
||||
throw new NotSupportedException(
|
||||
"Reading Exif data is not supported by this PHP installation."
|
||||
);
|
||||
}
|
||||
|
||||
$key = $this->argument(0)->value();
|
||||
|
||||
// try to read exif data from image file
|
||||
try {
|
||||
if ($image->dirname && $image->basename) {
|
||||
$stream = $image->dirname . '/' . $image->basename;
|
||||
} elseif (version_compare(PHP_VERSION, '7.2.0', '>=')) {
|
||||
// https://www.php.net/manual/en/function.exif-read-data.php#refsect1-function.exif-read-data-changelog
|
||||
$stream = $image->stream()->detach();
|
||||
} else {
|
||||
// https://bugs.php.net/bug.php?id=65187
|
||||
$stream = $image->encode('data-url')->encoded;
|
||||
}
|
||||
|
||||
$data = @exif_read_data($stream);
|
||||
|
||||
if (!is_null($key) && is_array($data)) {
|
||||
$data = array_key_exists($key, $data) ? $data[$key] : false;
|
||||
}
|
||||
|
||||
} catch (\Exception $e) {
|
||||
throw new NotReadableException(
|
||||
sprintf(
|
||||
"Cannot read the Exif data from the filename (%s) provided ",
|
||||
$image->dirname . '/' . $image->basename
|
||||
),
|
||||
$e->getCode(),
|
||||
$e
|
||||
);
|
||||
}
|
||||
|
||||
$this->setOutput($data);
|
||||
return true;
|
||||
}
|
||||
}
|
68
vendor/intervention/image/src/Intervention/Image/Commands/IptcCommand.php
vendored
Normal file
68
vendor/intervention/image/src/Intervention/Image/Commands/IptcCommand.php
vendored
Normal file
@ -0,0 +1,68 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
class IptcCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Read Iptc data from the given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
if ( ! function_exists('iptcparse')) {
|
||||
throw new NotSupportedException(
|
||||
"Reading Iptc data is not supported by this PHP installation."
|
||||
);
|
||||
}
|
||||
|
||||
$key = $this->argument(0)->value();
|
||||
|
||||
$info = [];
|
||||
@getimagesize($image->dirname .'/'. $image->basename, $info);
|
||||
|
||||
$data = [];
|
||||
|
||||
if (array_key_exists('APP13', $info)) {
|
||||
$iptc = iptcparse($info['APP13']);
|
||||
|
||||
if (is_array($iptc)) {
|
||||
$data['DocumentTitle'] = isset($iptc["2#005"][0]) ? $iptc["2#005"][0] : null;
|
||||
$data['Urgency'] = isset($iptc["2#010"][0]) ? $iptc["2#010"][0] : null;
|
||||
$data['Category'] = isset($iptc["2#015"][0]) ? $iptc["2#015"][0] : null;
|
||||
$data['Subcategories'] = isset($iptc["2#020"][0]) ? $iptc["2#020"][0] : null;
|
||||
$data['Keywords'] = isset($iptc["2#025"][0]) ? $iptc["2#025"] : null;
|
||||
$data['ReleaseDate'] = isset($iptc["2#030"][0]) ? $iptc["2#030"][0] : null;
|
||||
$data['ReleaseTime'] = isset($iptc["2#035"][0]) ? $iptc["2#035"][0] : null;
|
||||
$data['SpecialInstructions'] = isset($iptc["2#040"][0]) ? $iptc["2#040"][0] : null;
|
||||
$data['CreationDate'] = isset($iptc["2#055"][0]) ? $iptc["2#055"][0] : null;
|
||||
$data['CreationTime'] = isset($iptc["2#060"][0]) ? $iptc["2#060"][0] : null;
|
||||
$data['AuthorByline'] = isset($iptc["2#080"][0]) ? $iptc["2#080"][0] : null;
|
||||
$data['AuthorTitle'] = isset($iptc["2#085"][0]) ? $iptc["2#085"][0] : null;
|
||||
$data['City'] = isset($iptc["2#090"][0]) ? $iptc["2#090"][0] : null;
|
||||
$data['SubLocation'] = isset($iptc["2#092"][0]) ? $iptc["2#092"][0] : null;
|
||||
$data['State'] = isset($iptc["2#095"][0]) ? $iptc["2#095"][0] : null;
|
||||
$data['Country'] = isset($iptc["2#101"][0]) ? $iptc["2#101"][0] : null;
|
||||
$data['OTR'] = isset($iptc["2#103"][0]) ? $iptc["2#103"][0] : null;
|
||||
$data['Headline'] = isset($iptc["2#105"][0]) ? $iptc["2#105"][0] : null;
|
||||
$data['Source'] = isset($iptc["2#110"][0]) ? $iptc["2#110"][0] : null;
|
||||
$data['PhotoSource'] = isset($iptc["2#115"][0]) ? $iptc["2#115"][0] : null;
|
||||
$data['Copyright'] = isset($iptc["2#116"][0]) ? $iptc["2#116"][0] : null;
|
||||
$data['Caption'] = isset($iptc["2#120"][0]) ? $iptc["2#120"][0] : null;
|
||||
$data['CaptionWriter'] = isset($iptc["2#122"][0]) ? $iptc["2#122"][0] : null;
|
||||
}
|
||||
}
|
||||
|
||||
if (! is_null($key) && is_array($data)) {
|
||||
$data = array_key_exists($key, $data) ? $data[$key] : false;
|
||||
}
|
||||
|
||||
$this->setOutput($data);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
36
vendor/intervention/image/src/Intervention/Image/Commands/LineCommand.php
vendored
Normal file
36
vendor/intervention/image/src/Intervention/Image/Commands/LineCommand.php
vendored
Normal file
@ -0,0 +1,36 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Closure;
|
||||
|
||||
class LineCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Draws line on given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$x1 = $this->argument(0)->type('numeric')->required()->value();
|
||||
$y1 = $this->argument(1)->type('numeric')->required()->value();
|
||||
$x2 = $this->argument(2)->type('numeric')->required()->value();
|
||||
$y2 = $this->argument(3)->type('numeric')->required()->value();
|
||||
$callback = $this->argument(4)->type('closure')->value();
|
||||
|
||||
$line_classname = sprintf('\Intervention\Image\%s\Shapes\LineShape',
|
||||
$image->getDriver()->getDriverName());
|
||||
|
||||
$line = new $line_classname($x2, $y2);
|
||||
|
||||
if ($callback instanceof Closure) {
|
||||
$callback($line);
|
||||
}
|
||||
|
||||
$line->applyToImage($image, $x1, $y1);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
48
vendor/intervention/image/src/Intervention/Image/Commands/OrientateCommand.php
vendored
Normal file
48
vendor/intervention/image/src/Intervention/Image/Commands/OrientateCommand.php
vendored
Normal file
@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
class OrientateCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Correct image orientation according to Exif data
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
switch ($image->exif('Orientation')) {
|
||||
|
||||
case 2:
|
||||
$image->flip();
|
||||
break;
|
||||
|
||||
case 3:
|
||||
$image->rotate(180);
|
||||
break;
|
||||
|
||||
case 4:
|
||||
$image->rotate(180)->flip();
|
||||
break;
|
||||
|
||||
case 5:
|
||||
$image->rotate(270)->flip();
|
||||
break;
|
||||
|
||||
case 6:
|
||||
$image->rotate(270);
|
||||
break;
|
||||
|
||||
case 7:
|
||||
$image->rotate(90)->flip();
|
||||
break;
|
||||
|
||||
case 8:
|
||||
$image->rotate(90);
|
||||
break;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
49
vendor/intervention/image/src/Intervention/Image/Commands/PolygonCommand.php
vendored
Normal file
49
vendor/intervention/image/src/Intervention/Image/Commands/PolygonCommand.php
vendored
Normal file
@ -0,0 +1,49 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Closure;
|
||||
use Intervention\Image\Exception\InvalidArgumentException;
|
||||
|
||||
class PolygonCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Draw a polygon on given image
|
||||
*
|
||||
* @param \Intervention\Image\image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$points = $this->argument(0)->type('array')->required()->value();
|
||||
$callback = $this->argument(1)->type('closure')->value();
|
||||
|
||||
$vertices_count = count($points);
|
||||
|
||||
// check if number if coordinates is even
|
||||
if ($vertices_count % 2 !== 0) {
|
||||
throw new InvalidArgumentException(
|
||||
"The number of given polygon vertices must be even."
|
||||
);
|
||||
}
|
||||
|
||||
if ($vertices_count < 6) {
|
||||
throw new InvalidArgumentException(
|
||||
"You must have at least 3 points in your array."
|
||||
);
|
||||
}
|
||||
|
||||
$polygon_classname = sprintf('\Intervention\Image\%s\Shapes\PolygonShape',
|
||||
$image->getDriver()->getDriverName());
|
||||
|
||||
$polygon = new $polygon_classname($points);
|
||||
|
||||
if ($callback instanceof Closure) {
|
||||
$callback($polygon);
|
||||
}
|
||||
|
||||
$polygon->applyToImage($image);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
45
vendor/intervention/image/src/Intervention/Image/Commands/PsrResponseCommand.php
vendored
Normal file
45
vendor/intervention/image/src/Intervention/Image/Commands/PsrResponseCommand.php
vendored
Normal file
@ -0,0 +1,45 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use GuzzleHttp\Psr7\Response;
|
||||
|
||||
class PsrResponseCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Builds PSR7 compatible response. May replace "response" command in
|
||||
* some future.
|
||||
*
|
||||
* Method will generate binary stream and put it inside PSR-7
|
||||
* ResponseInterface. Following code can be optimized using native php
|
||||
* streams and more "clean" streaming, however drivers has to be updated
|
||||
* first.
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$format = $this->argument(0)->value();
|
||||
$quality = $this->argument(1)->between(0, 100)->value();
|
||||
|
||||
//Encoded property will be populated at this moment
|
||||
$stream = $image->stream($format, $quality);
|
||||
|
||||
$mimetype = finfo_buffer(
|
||||
finfo_open(FILEINFO_MIME_TYPE),
|
||||
$image->getEncoded()
|
||||
);
|
||||
|
||||
$this->setOutput(new Response(
|
||||
200,
|
||||
[
|
||||
'Content-Type' => $mimetype,
|
||||
'Content-Length' => strlen($image->getEncoded())
|
||||
],
|
||||
$stream
|
||||
));
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
36
vendor/intervention/image/src/Intervention/Image/Commands/RectangleCommand.php
vendored
Normal file
36
vendor/intervention/image/src/Intervention/Image/Commands/RectangleCommand.php
vendored
Normal file
@ -0,0 +1,36 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Closure;
|
||||
|
||||
class RectangleCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Draws rectangle on given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$x1 = $this->argument(0)->type('numeric')->required()->value();
|
||||
$y1 = $this->argument(1)->type('numeric')->required()->value();
|
||||
$x2 = $this->argument(2)->type('numeric')->required()->value();
|
||||
$y2 = $this->argument(3)->type('numeric')->required()->value();
|
||||
$callback = $this->argument(4)->type('closure')->value();
|
||||
|
||||
$rectangle_classname = sprintf('\Intervention\Image\%s\Shapes\RectangleShape',
|
||||
$image->getDriver()->getDriverName());
|
||||
|
||||
$rectangle = new $rectangle_classname($x1, $y1, $x2, $y2);
|
||||
|
||||
if ($callback instanceof Closure) {
|
||||
$callback($rectangle);
|
||||
}
|
||||
|
||||
$rectangle->applyToImage($image, $x1, $y1);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
26
vendor/intervention/image/src/Intervention/Image/Commands/ResponseCommand.php
vendored
Normal file
26
vendor/intervention/image/src/Intervention/Image/Commands/ResponseCommand.php
vendored
Normal file
@ -0,0 +1,26 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Intervention\Image\Response;
|
||||
|
||||
class ResponseCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Builds HTTP response from given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$format = $this->argument(0)->value();
|
||||
$quality = $this->argument(1)->between(0, 100)->value();
|
||||
|
||||
$response = new Response($image, $format, $quality);
|
||||
|
||||
$this->setOutput($response->make());
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
39
vendor/intervention/image/src/Intervention/Image/Commands/StreamCommand.php
vendored
Normal file
39
vendor/intervention/image/src/Intervention/Image/Commands/StreamCommand.php
vendored
Normal file
@ -0,0 +1,39 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
class StreamCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Builds PSR7 stream based on image data. Method uses Guzzle PSR7
|
||||
* implementation as easiest choice.
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$format = $this->argument(0)->value();
|
||||
$quality = $this->argument(1)->between(0, 100)->value();
|
||||
$data = $image->encode($format, $quality)->getEncoded();
|
||||
|
||||
$this->setOutput($this->getStream($data));
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Create stream from given data
|
||||
*
|
||||
* @param string $data
|
||||
* @return \Psr\Http\Message\StreamInterface
|
||||
*/
|
||||
protected function getStream($data)
|
||||
{
|
||||
if (class_exists(\GuzzleHttp\Psr7\Utils::class)) {
|
||||
return \GuzzleHttp\Psr7\Utils::streamFor($data); // guzzlehttp/psr7 >= 2.0
|
||||
}
|
||||
|
||||
return \GuzzleHttp\Psr7\stream_for($data); // guzzlehttp/psr7 < 2.0
|
||||
}
|
||||
}
|
34
vendor/intervention/image/src/Intervention/Image/Commands/TextCommand.php
vendored
Normal file
34
vendor/intervention/image/src/Intervention/Image/Commands/TextCommand.php
vendored
Normal file
@ -0,0 +1,34 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Commands;
|
||||
|
||||
use Closure;
|
||||
|
||||
class TextCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Write text on given image
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$text = $this->argument(0)->required()->value();
|
||||
$x = $this->argument(1)->type('numeric')->value(0);
|
||||
$y = $this->argument(2)->type('numeric')->value(0);
|
||||
$callback = $this->argument(3)->type('closure')->value();
|
||||
|
||||
$fontclassname = sprintf('\Intervention\Image\%s\Font',
|
||||
$image->getDriver()->getDriverName());
|
||||
|
||||
$font = new $fontclassname($text);
|
||||
|
||||
if ($callback instanceof Closure) {
|
||||
$callback($font);
|
||||
}
|
||||
|
||||
$font->applyToImage($image, $x, $y);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
92
vendor/intervention/image/src/Intervention/Image/Constraint.php
vendored
Normal file
92
vendor/intervention/image/src/Intervention/Image/Constraint.php
vendored
Normal file
@ -0,0 +1,92 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
class Constraint
|
||||
{
|
||||
/**
|
||||
* Bit value of aspect ratio constraint
|
||||
*/
|
||||
const ASPECTRATIO = 1;
|
||||
|
||||
/**
|
||||
* Bit value of upsize constraint
|
||||
*/
|
||||
const UPSIZE = 2;
|
||||
|
||||
/**
|
||||
* Constraint size
|
||||
*
|
||||
* @var \Intervention\Image\Size
|
||||
*/
|
||||
private $size;
|
||||
|
||||
/**
|
||||
* Integer value of fixed parameters
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
private $fixed = 0;
|
||||
|
||||
/**
|
||||
* Create a new constraint based on size
|
||||
*
|
||||
* @param Size $size
|
||||
*/
|
||||
public function __construct(Size $size)
|
||||
{
|
||||
$this->size = $size;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns current size of constraint
|
||||
*
|
||||
* @return \Intervention\Image\Size
|
||||
*/
|
||||
public function getSize()
|
||||
{
|
||||
return $this->size;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fix the given argument in current constraint
|
||||
*
|
||||
* @param int $type
|
||||
* @return void
|
||||
*/
|
||||
public function fix($type)
|
||||
{
|
||||
$this->fixed = ($this->fixed & ~(1 << $type)) | (1 << $type);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if given argument is fixed in current constraint
|
||||
*
|
||||
* @param int $type
|
||||
* @return boolean
|
||||
*/
|
||||
public function isFixed($type)
|
||||
{
|
||||
return (bool) ($this->fixed & (1 << $type));
|
||||
}
|
||||
|
||||
/**
|
||||
* Fixes aspect ratio in current constraint
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function aspectRatio()
|
||||
{
|
||||
$this->fix(self::ASPECTRATIO);
|
||||
}
|
||||
|
||||
/**
|
||||
* Fixes possibility to size up in current constraint
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function upsize()
|
||||
{
|
||||
$this->fix(self::UPSIZE);
|
||||
}
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/ImageException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/ImageException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class ImageException extends \RuntimeException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/InvalidArgumentException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/InvalidArgumentException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class InvalidArgumentException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/MissingDependencyException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/MissingDependencyException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class MissingDependencyException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/NotFoundException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/NotFoundException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class NotFoundException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/NotReadableException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/NotReadableException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class NotReadableException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/NotSupportedException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/NotSupportedException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class NotSupportedException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/NotWritableException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/NotWritableException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class NotWritableException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
8
vendor/intervention/image/src/Intervention/Image/Exception/RuntimeException.php
vendored
Normal file
8
vendor/intervention/image/src/Intervention/Image/Exception/RuntimeException.php
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Exception;
|
||||
|
||||
class RuntimeException extends ImageException
|
||||
{
|
||||
# nothing to override
|
||||
}
|
19
vendor/intervention/image/src/Intervention/Image/Facades/Image.php
vendored
Normal file
19
vendor/intervention/image/src/Intervention/Image/Facades/Image.php
vendored
Normal file
@ -0,0 +1,19 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Facades;
|
||||
|
||||
use Illuminate\Support\Facades\Facade;
|
||||
|
||||
/**
|
||||
* @method static \Intervention\Image\Image make(mixed $data)
|
||||
* @method static self configure(array $config)
|
||||
* @method static \Intervention\Image\Image canvas(int $width, int $height, mixed $background = null)
|
||||
* @method static \Intervention\Image\Image cache(\Closure $callback, int $lifetime = null, boolean $returnObj = false)
|
||||
*/
|
||||
class Image extends Facade
|
||||
{
|
||||
protected static function getFacadeAccessor()
|
||||
{
|
||||
return 'image';
|
||||
}
|
||||
}
|
92
vendor/intervention/image/src/Intervention/Image/File.php
vendored
Normal file
92
vendor/intervention/image/src/Intervention/Image/File.php
vendored
Normal file
@ -0,0 +1,92 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
class File
|
||||
{
|
||||
/**
|
||||
* Mime type
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $mime;
|
||||
|
||||
/**
|
||||
* Name of directory path
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $dirname;
|
||||
|
||||
/**
|
||||
* Basename of current file
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $basename;
|
||||
|
||||
/**
|
||||
* File extension of current file
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $extension;
|
||||
|
||||
/**
|
||||
* File name of current file
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $filename;
|
||||
|
||||
/**
|
||||
* Sets all instance properties from given path
|
||||
*
|
||||
* @param string $path
|
||||
*/
|
||||
public function setFileInfoFromPath($path)
|
||||
{
|
||||
$info = pathinfo($path);
|
||||
$this->dirname = array_key_exists('dirname', $info) ? $info['dirname'] : null;
|
||||
$this->basename = array_key_exists('basename', $info) ? $info['basename'] : null;
|
||||
$this->extension = array_key_exists('extension', $info) ? $info['extension'] : null;
|
||||
$this->filename = array_key_exists('filename', $info) ? $info['filename'] : null;
|
||||
|
||||
if (file_exists($path) && is_file($path)) {
|
||||
$this->mime = finfo_file(finfo_open(FILEINFO_MIME_TYPE), $path);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get file size
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function filesize()
|
||||
{
|
||||
$path = $this->basePath();
|
||||
|
||||
if (file_exists($path) && is_file($path)) {
|
||||
return filesize($path);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get fully qualified path
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function basePath()
|
||||
{
|
||||
if ($this->dirname && $this->basename) {
|
||||
return ($this->dirname .'/'. $this->basename);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
}
|
44
vendor/intervention/image/src/Intervention/Image/Filters/DemoFilter.php
vendored
Normal file
44
vendor/intervention/image/src/Intervention/Image/Filters/DemoFilter.php
vendored
Normal file
@ -0,0 +1,44 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Filters;
|
||||
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class DemoFilter implements FilterInterface
|
||||
{
|
||||
/**
|
||||
* Default size of filter effects
|
||||
*/
|
||||
const DEFAULT_SIZE = 10;
|
||||
|
||||
/**
|
||||
* Size of filter effects
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
private $size;
|
||||
|
||||
/**
|
||||
* Creates new instance of filter
|
||||
*
|
||||
* @param int $size
|
||||
*/
|
||||
public function __construct($size = null)
|
||||
{
|
||||
$this->size = is_numeric($size) ? intval($size) : self::DEFAULT_SIZE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Applies filter effects to given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function applyFilter(Image $image)
|
||||
{
|
||||
$image->pixelate($this->size);
|
||||
$image->greyscale();
|
||||
|
||||
return $image;
|
||||
}
|
||||
}
|
16
vendor/intervention/image/src/Intervention/Image/Filters/FilterInterface.php
vendored
Normal file
16
vendor/intervention/image/src/Intervention/Image/Filters/FilterInterface.php
vendored
Normal file
@ -0,0 +1,16 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Filters;
|
||||
|
||||
use Intervention\Image\Image;
|
||||
|
||||
interface FilterInterface
|
||||
{
|
||||
/**
|
||||
* Applies filter to given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function applyFilter(Image $image);
|
||||
}
|
227
vendor/intervention/image/src/Intervention/Image/Gd/Color.php
vendored
Normal file
227
vendor/intervention/image/src/Intervention/Image/Gd/Color.php
vendored
Normal file
@ -0,0 +1,227 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd;
|
||||
|
||||
use Intervention\Image\AbstractColor;
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
class Color extends AbstractColor
|
||||
{
|
||||
/**
|
||||
* RGB Red value of current color instance
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $r;
|
||||
|
||||
/**
|
||||
* RGB Green value of current color instance
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $g;
|
||||
|
||||
/**
|
||||
* RGB Blue value of current color instance
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $b;
|
||||
|
||||
/**
|
||||
* RGB Alpha value of current color instance
|
||||
*
|
||||
* @var float
|
||||
*/
|
||||
public $a;
|
||||
|
||||
/**
|
||||
* Initiates color object from integer
|
||||
*
|
||||
* @param int $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function initFromInteger($value)
|
||||
{
|
||||
$this->a = ($value >> 24) & 0xFF;
|
||||
$this->r = ($value >> 16) & 0xFF;
|
||||
$this->g = ($value >> 8) & 0xFF;
|
||||
$this->b = $value & 0xFF;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates color object from given array
|
||||
*
|
||||
* @param array $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function initFromArray($array)
|
||||
{
|
||||
$array = array_values($array);
|
||||
|
||||
if (count($array) == 4) {
|
||||
|
||||
// color array with alpha value
|
||||
list($r, $g, $b, $a) = $array;
|
||||
$this->a = $this->alpha2gd($a);
|
||||
|
||||
} elseif (count($array) == 3) {
|
||||
|
||||
// color array without alpha value
|
||||
list($r, $g, $b) = $array;
|
||||
$this->a = 0;
|
||||
|
||||
}
|
||||
|
||||
$this->r = $r;
|
||||
$this->g = $g;
|
||||
$this->b = $b;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates color object from given string
|
||||
*
|
||||
* @param string $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function initFromString($value)
|
||||
{
|
||||
if ($color = $this->rgbaFromString($value)) {
|
||||
$this->r = $color[0];
|
||||
$this->g = $color[1];
|
||||
$this->b = $color[2];
|
||||
$this->a = $this->alpha2gd($color[3]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates color object from given R, G and B values
|
||||
*
|
||||
* @param int $r
|
||||
* @param int $g
|
||||
* @param int $b
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function initFromRgb($r, $g, $b)
|
||||
{
|
||||
$this->r = intval($r);
|
||||
$this->g = intval($g);
|
||||
$this->b = intval($b);
|
||||
$this->a = 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates color object from given R, G, B and A values
|
||||
*
|
||||
* @param int $r
|
||||
* @param int $g
|
||||
* @param int $b
|
||||
* @param float $a
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function initFromRgba($r, $g, $b, $a = 1)
|
||||
{
|
||||
$this->r = intval($r);
|
||||
$this->g = intval($g);
|
||||
$this->b = intval($b);
|
||||
$this->a = $this->alpha2gd($a);
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates color object from given ImagickPixel object
|
||||
*
|
||||
* @param ImagickPixel $value
|
||||
* @return \Intervention\Image\AbstractColor
|
||||
*/
|
||||
public function initFromObject($value)
|
||||
{
|
||||
throw new NotSupportedException(
|
||||
"GD colors cannot init from ImagickPixel objects."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates integer value of current color instance
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getInt()
|
||||
{
|
||||
return ($this->a << 24) + ($this->r << 16) + ($this->g << 8) + $this->b;
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates hexadecimal value of current color instance
|
||||
*
|
||||
* @param string $prefix
|
||||
* @return string
|
||||
*/
|
||||
public function getHex($prefix = '')
|
||||
{
|
||||
return sprintf('%s%02x%02x%02x', $prefix, $this->r, $this->g, $this->b);
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates RGB(A) in array format of current color instance
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function getArray()
|
||||
{
|
||||
return [$this->r, $this->g, $this->b, round(1 - $this->a / 127, 2)];
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates RGBA in string format of current color instance
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getRgba()
|
||||
{
|
||||
return sprintf('rgba(%d, %d, %d, %.2F)', $this->r, $this->g, $this->b, round(1 - $this->a / 127, 2));
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines if current color is different from given color
|
||||
*
|
||||
* @param AbstractColor $color
|
||||
* @param int $tolerance
|
||||
* @return boolean
|
||||
*/
|
||||
public function differs(AbstractColor $color, $tolerance = 0)
|
||||
{
|
||||
$color_tolerance = round($tolerance * 2.55);
|
||||
$alpha_tolerance = round($tolerance * 1.27);
|
||||
|
||||
$delta = [
|
||||
'r' => abs($color->r - $this->r),
|
||||
'g' => abs($color->g - $this->g),
|
||||
'b' => abs($color->b - $this->b),
|
||||
'a' => abs($color->a - $this->a)
|
||||
];
|
||||
|
||||
return (
|
||||
$delta['r'] > $color_tolerance or
|
||||
$delta['g'] > $color_tolerance or
|
||||
$delta['b'] > $color_tolerance or
|
||||
$delta['a'] > $alpha_tolerance
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert rgba alpha (0-1) value to gd value (0-127)
|
||||
*
|
||||
* @param float $input
|
||||
* @return int
|
||||
*/
|
||||
private function alpha2gd($input)
|
||||
{
|
||||
$oldMin = 0;
|
||||
$oldMax = 1;
|
||||
|
||||
$newMin = 127;
|
||||
$newMax = 0;
|
||||
|
||||
return ceil(((($input- $oldMin) * ($newMax - $newMin)) / ($oldMax - $oldMin)) + $newMin);
|
||||
}
|
||||
}
|
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/BackupCommand.php
vendored
Normal file
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/BackupCommand.php
vendored
Normal file
@ -0,0 +1,25 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class BackupCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Saves a backups of current state of image core
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$backupName = $this->argument(0)->value();
|
||||
|
||||
// clone current image resource
|
||||
$clone = clone $image;
|
||||
$image->setBackup($clone->getCore(), $backupName);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/BlurCommand.php
vendored
Normal file
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/BlurCommand.php
vendored
Normal file
@ -0,0 +1,25 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class BlurCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Applies blur effect on image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$amount = $this->argument(0)->between(0, 100)->value(1);
|
||||
|
||||
for ($i=0; $i < intval($amount); $i++) {
|
||||
imagefilter($image->getCore(), IMG_FILTER_GAUSSIAN_BLUR);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/BrightnessCommand.php
vendored
Normal file
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/BrightnessCommand.php
vendored
Normal file
@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class BrightnessCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Changes image brightness
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$level = $this->argument(0)->between(-100, 100)->required()->value();
|
||||
|
||||
return imagefilter($image->getCore(), IMG_FILTER_BRIGHTNESS, ($level * 2.55));
|
||||
}
|
||||
}
|
29
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ColorizeCommand.php
vendored
Normal file
29
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ColorizeCommand.php
vendored
Normal file
@ -0,0 +1,29 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class ColorizeCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Changes balance of different RGB color channels
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$red = $this->argument(0)->between(-100, 100)->required()->value();
|
||||
$green = $this->argument(1)->between(-100, 100)->required()->value();
|
||||
$blue = $this->argument(2)->between(-100, 100)->required()->value();
|
||||
|
||||
// normalize colorize levels
|
||||
$red = round($red * 2.55);
|
||||
$green = round($green * 2.55);
|
||||
$blue = round($blue * 2.55);
|
||||
|
||||
// apply filter
|
||||
return imagefilter($image->getCore(), IMG_FILTER_COLORIZE, $red, $green, $blue);
|
||||
}
|
||||
}
|
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ContrastCommand.php
vendored
Normal file
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ContrastCommand.php
vendored
Normal file
@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class ContrastCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Changes contrast of image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$level = $this->argument(0)->between(-100, 100)->required()->value();
|
||||
|
||||
return imagefilter($image->getCore(), IMG_FILTER_CONTRAST, ($level * -1));
|
||||
}
|
||||
}
|
40
vendor/intervention/image/src/Intervention/Image/Gd/Commands/CropCommand.php
vendored
Normal file
40
vendor/intervention/image/src/Intervention/Image/Gd/Commands/CropCommand.php
vendored
Normal file
@ -0,0 +1,40 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Point;
|
||||
use Intervention\Image\Size;
|
||||
|
||||
class CropCommand extends ResizeCommand
|
||||
{
|
||||
/**
|
||||
* Crop an image instance
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$width = $this->argument(0)->type('digit')->required()->value();
|
||||
$height = $this->argument(1)->type('digit')->required()->value();
|
||||
$x = $this->argument(2)->type('digit')->value();
|
||||
$y = $this->argument(3)->type('digit')->value();
|
||||
|
||||
if (is_null($width) || is_null($height)) {
|
||||
throw new \Intervention\Image\Exception\InvalidArgumentException(
|
||||
"Width and height of cutout needs to be defined."
|
||||
);
|
||||
}
|
||||
|
||||
$cropped = new Size($width, $height);
|
||||
$position = new Point($x, $y);
|
||||
|
||||
// align boxes
|
||||
if (is_null($x) && is_null($y)) {
|
||||
$position = $image->getSize()->align('center')->relativePosition($cropped->align('center'));
|
||||
}
|
||||
|
||||
// crop image core
|
||||
return $this->modify($image, 0, 0, $position->x, $position->y, $cropped->width, $cropped->height, $cropped->width, $cropped->height);
|
||||
}
|
||||
}
|
27
vendor/intervention/image/src/Intervention/Image/Gd/Commands/DestroyCommand.php
vendored
Normal file
27
vendor/intervention/image/src/Intervention/Image/Gd/Commands/DestroyCommand.php
vendored
Normal file
@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class DestroyCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Destroys current image core and frees up memory
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
// destroy image core
|
||||
imagedestroy($image->getCore());
|
||||
|
||||
// destroy backups
|
||||
foreach ($image->getBackups() as $backup) {
|
||||
imagedestroy($backup);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
69
vendor/intervention/image/src/Intervention/Image/Gd/Commands/FillCommand.php
vendored
Normal file
69
vendor/intervention/image/src/Intervention/Image/Gd/Commands/FillCommand.php
vendored
Normal file
@ -0,0 +1,69 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
use Intervention\Image\Gd\Color;
|
||||
use Intervention\Image\Gd\Decoder;
|
||||
|
||||
class FillCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Fills image with color or pattern
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$filling = $this->argument(0)->value();
|
||||
$x = $this->argument(1)->type('digit')->value();
|
||||
$y = $this->argument(2)->type('digit')->value();
|
||||
|
||||
$width = $image->getWidth();
|
||||
$height = $image->getHeight();
|
||||
$resource = $image->getCore();
|
||||
|
||||
try {
|
||||
|
||||
// set image tile filling
|
||||
$source = new Decoder;
|
||||
$tile = $source->init($filling);
|
||||
imagesettile($image->getCore(), $tile->getCore());
|
||||
$filling = IMG_COLOR_TILED;
|
||||
|
||||
} catch (\Intervention\Image\Exception\NotReadableException $e) {
|
||||
|
||||
// set solid color filling
|
||||
$color = new Color($filling);
|
||||
$filling = $color->getInt();
|
||||
}
|
||||
|
||||
imagealphablending($resource, true);
|
||||
|
||||
if (is_int($x) && is_int($y)) {
|
||||
|
||||
// resource should be visible through transparency
|
||||
$base = $image->getDriver()->newImage($width, $height)->getCore();
|
||||
imagecopy($base, $resource, 0, 0, 0, 0, $width, $height);
|
||||
|
||||
// floodfill if exact position is defined
|
||||
imagefill($resource, $x, $y, $filling);
|
||||
|
||||
// copy filled original over base
|
||||
imagecopy($base, $resource, 0, 0, 0, 0, $width, $height);
|
||||
|
||||
// set base as new resource-core
|
||||
$image->setCore($base);
|
||||
imagedestroy($resource);
|
||||
|
||||
} else {
|
||||
// fill whole image otherwise
|
||||
imagefilledrectangle($resource, 0, 0, $width - 1, $height - 1, $filling);
|
||||
}
|
||||
|
||||
isset($tile) ? imagedestroy($tile->getCore()) : null;
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
32
vendor/intervention/image/src/Intervention/Image/Gd/Commands/FitCommand.php
vendored
Normal file
32
vendor/intervention/image/src/Intervention/Image/Gd/Commands/FitCommand.php
vendored
Normal file
@ -0,0 +1,32 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Size;
|
||||
|
||||
class FitCommand extends ResizeCommand
|
||||
{
|
||||
/**
|
||||
* Crops and resized an image at the same time
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$width = $this->argument(0)->type('digit')->required()->value();
|
||||
$height = $this->argument(1)->type('digit')->value($width);
|
||||
$constraints = $this->argument(2)->type('closure')->value();
|
||||
$position = $this->argument(3)->type('string')->value('center');
|
||||
|
||||
// calculate size
|
||||
$cropped = $image->getSize()->fit(new Size($width, $height), $position);
|
||||
$resized = clone $cropped;
|
||||
$resized = $resized->resize($width, $height, $constraints);
|
||||
|
||||
// modify image
|
||||
$this->modify($image, 0, 0, $cropped->pivot->x, $cropped->pivot->y, $resized->getWidth(), $resized->getHeight(), $cropped->getWidth(), $cropped->getHeight());
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
37
vendor/intervention/image/src/Intervention/Image/Gd/Commands/FlipCommand.php
vendored
Normal file
37
vendor/intervention/image/src/Intervention/Image/Gd/Commands/FlipCommand.php
vendored
Normal file
@ -0,0 +1,37 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
class FlipCommand extends ResizeCommand
|
||||
{
|
||||
/**
|
||||
* Mirrors an image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$mode = $this->argument(0)->value('h');
|
||||
|
||||
$size = $image->getSize();
|
||||
$dst = clone $size;
|
||||
|
||||
switch (strtolower($mode)) {
|
||||
case 2:
|
||||
case 'v':
|
||||
case 'vert':
|
||||
case 'vertical':
|
||||
$size->pivot->y = $size->height - 1;
|
||||
$size->height = $size->height * (-1);
|
||||
break;
|
||||
|
||||
default:
|
||||
$size->pivot->x = $size->width - 1;
|
||||
$size->width = $size->width * (-1);
|
||||
break;
|
||||
}
|
||||
|
||||
return $this->modify($image, 0, 0, $size->pivot->x, $size->pivot->y, $dst->width, $dst->height, $size->width, $size->height);
|
||||
}
|
||||
}
|
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/GammaCommand.php
vendored
Normal file
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/GammaCommand.php
vendored
Normal file
@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class GammaCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Applies gamma correction to a given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$gamma = $this->argument(0)->type('numeric')->required()->value();
|
||||
|
||||
return imagegammacorrect($image->getCore(), 1, $gamma);
|
||||
}
|
||||
}
|
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/GetSizeCommand.php
vendored
Normal file
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/GetSizeCommand.php
vendored
Normal file
@ -0,0 +1,25 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
use Intervention\Image\Size;
|
||||
|
||||
class GetSizeCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Reads size of given image instance in pixels
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$this->setOutput(new Size(
|
||||
imagesx($image->getCore()),
|
||||
imagesy($image->getCore())
|
||||
));
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
19
vendor/intervention/image/src/Intervention/Image/Gd/Commands/GreyscaleCommand.php
vendored
Normal file
19
vendor/intervention/image/src/Intervention/Image/Gd/Commands/GreyscaleCommand.php
vendored
Normal file
@ -0,0 +1,19 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class GreyscaleCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Turns an image into a greyscale version
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
return imagefilter($image->getCore(), IMG_FILTER_GRAYSCALE);
|
||||
}
|
||||
}
|
28
vendor/intervention/image/src/Intervention/Image/Gd/Commands/HeightenCommand.php
vendored
Normal file
28
vendor/intervention/image/src/Intervention/Image/Gd/Commands/HeightenCommand.php
vendored
Normal file
@ -0,0 +1,28 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
class HeightenCommand extends ResizeCommand
|
||||
{
|
||||
/**
|
||||
* Resize image proportionally to given height
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$height = $this->argument(0)->type('digit')->required()->value();
|
||||
$additionalConstraints = $this->argument(1)->type('closure')->value();
|
||||
|
||||
$this->arguments[0] = null;
|
||||
$this->arguments[1] = $height;
|
||||
$this->arguments[2] = function ($constraint) use ($additionalConstraints) {
|
||||
$constraint->aspectRatio();
|
||||
if(is_callable($additionalConstraints))
|
||||
$additionalConstraints($constraint);
|
||||
};
|
||||
|
||||
return parent::execute($image);
|
||||
}
|
||||
}
|
34
vendor/intervention/image/src/Intervention/Image/Gd/Commands/InsertCommand.php
vendored
Normal file
34
vendor/intervention/image/src/Intervention/Image/Gd/Commands/InsertCommand.php
vendored
Normal file
@ -0,0 +1,34 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class InsertCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Insert another image into given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$source = $this->argument(0)->required()->value();
|
||||
$position = $this->argument(1)->type('string')->value();
|
||||
$x = $this->argument(2)->type('digit')->value(0);
|
||||
$y = $this->argument(3)->type('digit')->value(0);
|
||||
|
||||
// build watermark
|
||||
$watermark = $image->getDriver()->init($source);
|
||||
|
||||
// define insertion point
|
||||
$image_size = $image->getSize()->align($position, $x, $y);
|
||||
$watermark_size = $watermark->getSize()->align($position);
|
||||
$target = $image_size->relativePosition($watermark_size);
|
||||
|
||||
// insert image at position
|
||||
imagealphablending($image->getCore(), true);
|
||||
return imagecopy($image->getCore(), $watermark->getCore(), $target->x, $target->y, 0, 0, $watermark_size->width, $watermark_size->height);
|
||||
}
|
||||
}
|
23
vendor/intervention/image/src/Intervention/Image/Gd/Commands/InterlaceCommand.php
vendored
Normal file
23
vendor/intervention/image/src/Intervention/Image/Gd/Commands/InterlaceCommand.php
vendored
Normal file
@ -0,0 +1,23 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class InterlaceCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Toggles interlaced encoding mode
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$mode = $this->argument(0)->type('bool')->value(true);
|
||||
|
||||
imageinterlace($image->getCore(), $mode);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
19
vendor/intervention/image/src/Intervention/Image/Gd/Commands/InvertCommand.php
vendored
Normal file
19
vendor/intervention/image/src/Intervention/Image/Gd/Commands/InvertCommand.php
vendored
Normal file
@ -0,0 +1,19 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class InvertCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Inverts colors of an image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
return imagefilter($image->getCore(), IMG_FILTER_NEGATE);
|
||||
}
|
||||
}
|
53
vendor/intervention/image/src/Intervention/Image/Gd/Commands/LimitColorsCommand.php
vendored
Normal file
53
vendor/intervention/image/src/Intervention/Image/Gd/Commands/LimitColorsCommand.php
vendored
Normal file
@ -0,0 +1,53 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
|
||||
class LimitColorsCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Reduces colors of a given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$count = $this->argument(0)->value();
|
||||
$matte = $this->argument(1)->value();
|
||||
|
||||
// get current image size
|
||||
$size = $image->getSize();
|
||||
|
||||
// create empty canvas
|
||||
$resource = imagecreatetruecolor($size->width, $size->height);
|
||||
|
||||
// define matte
|
||||
if (is_null($matte)) {
|
||||
$matte = imagecolorallocatealpha($resource, 255, 255, 255, 127);
|
||||
} else {
|
||||
$matte = $image->getDriver()->parseColor($matte)->getInt();
|
||||
}
|
||||
|
||||
// fill with matte and copy original image
|
||||
imagefill($resource, 0, 0, $matte);
|
||||
|
||||
// set transparency
|
||||
imagecolortransparent($resource, $matte);
|
||||
|
||||
// copy original image
|
||||
imagecopy($resource, $image->getCore(), 0, 0, 0, 0, $size->width, $size->height);
|
||||
|
||||
if (is_numeric($count) && $count <= 256) {
|
||||
// decrease colors
|
||||
imagetruecolortopalette($resource, true, $count);
|
||||
}
|
||||
|
||||
// set new resource
|
||||
$image->setCore($resource);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
83
vendor/intervention/image/src/Intervention/Image/Gd/Commands/MaskCommand.php
vendored
Normal file
83
vendor/intervention/image/src/Intervention/Image/Gd/Commands/MaskCommand.php
vendored
Normal file
@ -0,0 +1,83 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class MaskCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Applies an alpha mask to an image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$mask_source = $this->argument(0)->value();
|
||||
$mask_w_alpha = $this->argument(1)->type('bool')->value(false);
|
||||
|
||||
$image_size = $image->getSize();
|
||||
|
||||
// create empty canvas
|
||||
$canvas = $image->getDriver()->newImage($image_size->width, $image_size->height, [0,0,0,0]);
|
||||
|
||||
// build mask image from source
|
||||
$mask = $image->getDriver()->init($mask_source);
|
||||
$mask_size = $mask->getSize();
|
||||
|
||||
// resize mask to size of current image (if necessary)
|
||||
if ($mask_size != $image_size) {
|
||||
$mask->resize($image_size->width, $image_size->height);
|
||||
}
|
||||
|
||||
imagealphablending($canvas->getCore(), false);
|
||||
|
||||
if ( ! $mask_w_alpha) {
|
||||
// mask from greyscale image
|
||||
imagefilter($mask->getCore(), IMG_FILTER_GRAYSCALE);
|
||||
}
|
||||
|
||||
// redraw old image pixel by pixel considering alpha map
|
||||
for ($x=0; $x < $image_size->width; $x++) {
|
||||
for ($y=0; $y < $image_size->height; $y++) {
|
||||
|
||||
$color = $image->pickColor($x, $y, 'array');
|
||||
$alpha = $mask->pickColor($x, $y, 'array');
|
||||
|
||||
if ($mask_w_alpha) {
|
||||
$alpha = $alpha[3]; // use alpha channel as mask
|
||||
} else {
|
||||
|
||||
if ($alpha[3] == 0) { // transparent as black
|
||||
$alpha = 0;
|
||||
} else {
|
||||
|
||||
// $alpha = floatval(round((($alpha[0] + $alpha[1] + $alpha[3]) / 3) / 255, 2));
|
||||
|
||||
// image is greyscale, so channel doesn't matter (use red channel)
|
||||
$alpha = floatval(round($alpha[0] / 255, 2));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// preserve alpha of original image...
|
||||
if ($color[3] < $alpha) {
|
||||
$alpha = $color[3];
|
||||
}
|
||||
|
||||
// replace alpha value
|
||||
$color[3] = $alpha;
|
||||
|
||||
// redraw pixel
|
||||
$canvas->pixel($color, $x, $y);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// replace current image with masked instance
|
||||
$image->setCore($canvas->getCore());
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
31
vendor/intervention/image/src/Intervention/Image/Gd/Commands/OpacityCommand.php
vendored
Normal file
31
vendor/intervention/image/src/Intervention/Image/Gd/Commands/OpacityCommand.php
vendored
Normal file
@ -0,0 +1,31 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class OpacityCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Defines opacity of an image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$transparency = $this->argument(0)->between(0, 100)->required()->value();
|
||||
|
||||
// get size of image
|
||||
$size = $image->getSize();
|
||||
|
||||
// build temp alpha mask
|
||||
$mask_color = sprintf('rgba(0, 0, 0, %.1F)', $transparency / 100);
|
||||
$mask = $image->getDriver()->newImage($size->width, $size->height, $mask_color);
|
||||
|
||||
// mask image
|
||||
$image->mask($mask->getCore(), true);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
37
vendor/intervention/image/src/Intervention/Image/Gd/Commands/PickColorCommand.php
vendored
Normal file
37
vendor/intervention/image/src/Intervention/Image/Gd/Commands/PickColorCommand.php
vendored
Normal file
@ -0,0 +1,37 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
use Intervention\Image\Gd\Color;
|
||||
|
||||
class PickColorCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Read color information from a certain position
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$x = $this->argument(0)->type('digit')->required()->value();
|
||||
$y = $this->argument(1)->type('digit')->required()->value();
|
||||
$format = $this->argument(2)->type('string')->value('array');
|
||||
|
||||
// pick color
|
||||
$color = imagecolorat($image->getCore(), $x, $y);
|
||||
|
||||
if ( ! imageistruecolor($image->getCore())) {
|
||||
$color = imagecolorsforindex($image->getCore(), $color);
|
||||
$color['alpha'] = round(1 - $color['alpha'] / 127, 2);
|
||||
}
|
||||
|
||||
$color = new Color($color);
|
||||
|
||||
// format to output
|
||||
$this->setOutput($color->format($format));
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/PixelCommand.php
vendored
Normal file
25
vendor/intervention/image/src/Intervention/Image/Gd/Commands/PixelCommand.php
vendored
Normal file
@ -0,0 +1,25 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
use Intervention\Image\Gd\Color;
|
||||
|
||||
class PixelCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Draws one pixel to a given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$color = $this->argument(0)->required()->value();
|
||||
$color = new Color($color);
|
||||
$x = $this->argument(1)->type('digit')->required()->value();
|
||||
$y = $this->argument(2)->type('digit')->required()->value();
|
||||
|
||||
return imagesetpixel($image->getCore(), $x, $y, $color->getInt());
|
||||
}
|
||||
}
|
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/PixelateCommand.php
vendored
Normal file
21
vendor/intervention/image/src/Intervention/Image/Gd/Commands/PixelateCommand.php
vendored
Normal file
@ -0,0 +1,21 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class PixelateCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Applies a pixelation effect to a given image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$size = $this->argument(0)->type('digit')->value(10);
|
||||
|
||||
return imagefilter($image->getCore(), IMG_FILTER_PIXELATE, $size, true);
|
||||
}
|
||||
}
|
39
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ResetCommand.php
vendored
Normal file
39
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ResetCommand.php
vendored
Normal file
@ -0,0 +1,39 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
use Intervention\Image\Exception\RuntimeException;
|
||||
|
||||
class ResetCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Resets given image to its backup state
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$backupName = $this->argument(0)->value();
|
||||
$backup = $image->getBackup($backupName);
|
||||
|
||||
if (is_resource($backup) || $backup instanceof \GdImage) {
|
||||
|
||||
// destroy current resource
|
||||
imagedestroy($image->getCore());
|
||||
|
||||
// clone backup
|
||||
$backup = $image->getDriver()->cloneCore($backup);
|
||||
|
||||
// reset to new resource
|
||||
$image->setCore($backup);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
throw new RuntimeException(
|
||||
"Backup not available. Call backup() before reset()."
|
||||
);
|
||||
}
|
||||
}
|
83
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ResizeCanvasCommand.php
vendored
Normal file
83
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ResizeCanvasCommand.php
vendored
Normal file
@ -0,0 +1,83 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class ResizeCanvasCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Resizes image boundaries
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$width = $this->argument(0)->type('digit')->required()->value();
|
||||
$height = $this->argument(1)->type('digit')->required()->value();
|
||||
$anchor = $this->argument(2)->value('center');
|
||||
$relative = $this->argument(3)->type('boolean')->value(false);
|
||||
$bgcolor = $this->argument(4)->value();
|
||||
|
||||
$original_width = $image->getWidth();
|
||||
$original_height = $image->getHeight();
|
||||
|
||||
// check of only width or height is set
|
||||
$width = is_null($width) ? $original_width : intval($width);
|
||||
$height = is_null($height) ? $original_height : intval($height);
|
||||
|
||||
// check on relative width/height
|
||||
if ($relative) {
|
||||
$width = $original_width + $width;
|
||||
$height = $original_height + $height;
|
||||
}
|
||||
|
||||
// check for negative width/height
|
||||
$width = ($width <= 0) ? $width + $original_width : $width;
|
||||
$height = ($height <= 0) ? $height + $original_height : $height;
|
||||
|
||||
// create new canvas
|
||||
$canvas = $image->getDriver()->newImage($width, $height, $bgcolor);
|
||||
|
||||
// set copy position
|
||||
$canvas_size = $canvas->getSize()->align($anchor);
|
||||
$image_size = $image->getSize()->align($anchor);
|
||||
$canvas_pos = $image_size->relativePosition($canvas_size);
|
||||
$image_pos = $canvas_size->relativePosition($image_size);
|
||||
|
||||
if ($width <= $original_width) {
|
||||
$dst_x = 0;
|
||||
$src_x = $canvas_pos->x;
|
||||
$src_w = $canvas_size->width;
|
||||
} else {
|
||||
$dst_x = $image_pos->x;
|
||||
$src_x = 0;
|
||||
$src_w = $original_width;
|
||||
}
|
||||
|
||||
if ($height <= $original_height) {
|
||||
$dst_y = 0;
|
||||
$src_y = $canvas_pos->y;
|
||||
$src_h = $canvas_size->height;
|
||||
} else {
|
||||
$dst_y = $image_pos->y;
|
||||
$src_y = 0;
|
||||
$src_h = $original_height;
|
||||
}
|
||||
|
||||
// make image area transparent to keep transparency
|
||||
// even if background-color is set
|
||||
$transparent = imagecolorallocatealpha($canvas->getCore(), 255, 255, 255, 127);
|
||||
imagealphablending($canvas->getCore(), false); // do not blend / just overwrite
|
||||
imagefilledrectangle($canvas->getCore(), $dst_x, $dst_y, $dst_x + $src_w - 1, $dst_y + $src_h - 1, $transparent);
|
||||
|
||||
// copy image into new canvas
|
||||
imagecopy($canvas->getCore(), $image->getCore(), $dst_x, $dst_y, $src_x, $src_y, $src_w, $src_h);
|
||||
|
||||
// set new core to canvas
|
||||
$image->setCore($canvas->getCore());
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
84
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ResizeCommand.php
vendored
Normal file
84
vendor/intervention/image/src/Intervention/Image/Gd/Commands/ResizeCommand.php
vendored
Normal file
@ -0,0 +1,84 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class ResizeCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Resizes image dimensions
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$width = $this->argument(0)->value();
|
||||
$height = $this->argument(1)->value();
|
||||
$constraints = $this->argument(2)->type('closure')->value();
|
||||
|
||||
// resize box
|
||||
$resized = $image->getSize()->resize($width, $height, $constraints);
|
||||
|
||||
// modify image
|
||||
$this->modify($image, 0, 0, 0, 0, $resized->getWidth(), $resized->getHeight(), $image->getWidth(), $image->getHeight());
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Wrapper function for 'imagecopyresampled'
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $dst_x
|
||||
* @param int $dst_y
|
||||
* @param int $src_x
|
||||
* @param int $src_y
|
||||
* @param int $dst_w
|
||||
* @param int $dst_h
|
||||
* @param int $src_w
|
||||
* @param int $src_h
|
||||
* @return boolean
|
||||
*/
|
||||
protected function modify($image, $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h)
|
||||
{
|
||||
// create new image
|
||||
$modified = imagecreatetruecolor(intval($dst_w), intval($dst_h));
|
||||
|
||||
// get current image
|
||||
$resource = $image->getCore();
|
||||
|
||||
// preserve transparency
|
||||
$transIndex = imagecolortransparent($resource);
|
||||
|
||||
if ($transIndex != -1) {
|
||||
$rgba = imagecolorsforindex($modified, $transIndex);
|
||||
$transColor = imagecolorallocatealpha($modified, $rgba['red'], $rgba['green'], $rgba['blue'], 127);
|
||||
imagefill($modified, 0, 0, $transColor);
|
||||
imagecolortransparent($modified, $transColor);
|
||||
} else {
|
||||
imagealphablending($modified, false);
|
||||
imagesavealpha($modified, true);
|
||||
}
|
||||
|
||||
// copy content from resource
|
||||
$result = imagecopyresampled(
|
||||
$modified,
|
||||
$resource,
|
||||
$dst_x,
|
||||
$dst_y,
|
||||
$src_x,
|
||||
$src_y,
|
||||
intval($dst_w),
|
||||
intval($dst_h),
|
||||
$src_w,
|
||||
$src_h
|
||||
);
|
||||
|
||||
// set new content as recource
|
||||
$image->setCore($modified);
|
||||
|
||||
return $result;
|
||||
}
|
||||
}
|
30
vendor/intervention/image/src/Intervention/Image/Gd/Commands/RotateCommand.php
vendored
Normal file
30
vendor/intervention/image/src/Intervention/Image/Gd/Commands/RotateCommand.php
vendored
Normal file
@ -0,0 +1,30 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
use Intervention\Image\Gd\Color;
|
||||
|
||||
class RotateCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Rotates image counter clockwise
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$angle = $this->argument(0)->type('numeric')->required()->value();
|
||||
$color = $this->argument(1)->value();
|
||||
$color = new Color($color);
|
||||
|
||||
// restrict rotations beyond 360 degrees, since the end result is the same
|
||||
$angle = fmod($angle, 360);
|
||||
|
||||
// rotate image
|
||||
$image->setCore(imagerotate($image->getCore(), $angle, $color->getInt()));
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
34
vendor/intervention/image/src/Intervention/Image/Gd/Commands/SharpenCommand.php
vendored
Normal file
34
vendor/intervention/image/src/Intervention/Image/Gd/Commands/SharpenCommand.php
vendored
Normal file
@ -0,0 +1,34 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Commands\AbstractCommand;
|
||||
|
||||
class SharpenCommand extends AbstractCommand
|
||||
{
|
||||
/**
|
||||
* Sharpen image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$amount = $this->argument(0)->between(0, 100)->value(10);
|
||||
|
||||
// build matrix
|
||||
$min = $amount >= 10 ? $amount * -0.01 : 0;
|
||||
$max = $amount * -0.025;
|
||||
$abs = ((4 * $min + 4 * $max) * -1) + 1;
|
||||
$div = 1;
|
||||
|
||||
$matrix = [
|
||||
[$min, $max, $min],
|
||||
[$max, $abs, $max],
|
||||
[$min, $max, $min]
|
||||
];
|
||||
|
||||
// apply the matrix
|
||||
return imageconvolution($image->getCore(), $matrix, $div, 0);
|
||||
}
|
||||
}
|
176
vendor/intervention/image/src/Intervention/Image/Gd/Commands/TrimCommand.php
vendored
Normal file
176
vendor/intervention/image/src/Intervention/Image/Gd/Commands/TrimCommand.php
vendored
Normal file
@ -0,0 +1,176 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
use Intervention\Image\Gd\Color;
|
||||
|
||||
class TrimCommand extends ResizeCommand
|
||||
{
|
||||
/**
|
||||
* Trims away parts of an image
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$base = $this->argument(0)->type('string')->value();
|
||||
$away = $this->argument(1)->value();
|
||||
$tolerance = $this->argument(2)->type('numeric')->value(0);
|
||||
$feather = $this->argument(3)->type('numeric')->value(0);
|
||||
|
||||
$width = $image->getWidth();
|
||||
$height = $image->getHeight();
|
||||
|
||||
// default values
|
||||
$checkTransparency = false;
|
||||
|
||||
// define borders to trim away
|
||||
if (is_null($away)) {
|
||||
$away = ['top', 'right', 'bottom', 'left'];
|
||||
} elseif (is_string($away)) {
|
||||
$away = [$away];
|
||||
}
|
||||
|
||||
// lower border names
|
||||
foreach ($away as $key => $value) {
|
||||
$away[$key] = strtolower($value);
|
||||
}
|
||||
|
||||
// define base color position
|
||||
switch (strtolower($base)) {
|
||||
case 'transparent':
|
||||
case 'trans':
|
||||
$checkTransparency = true;
|
||||
$base_x = 0;
|
||||
$base_y = 0;
|
||||
break;
|
||||
|
||||
case 'bottom-right':
|
||||
case 'right-bottom':
|
||||
$base_x = $width - 1;
|
||||
$base_y = $height - 1;
|
||||
break;
|
||||
|
||||
default:
|
||||
case 'top-left':
|
||||
case 'left-top':
|
||||
$base_x = 0;
|
||||
$base_y = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
// pick base color
|
||||
if ($checkTransparency) {
|
||||
$color = new Color; // color will only be used to compare alpha channel
|
||||
} else {
|
||||
$color = $image->pickColor($base_x, $base_y, 'object');
|
||||
}
|
||||
|
||||
$top_x = 0;
|
||||
$top_y = 0;
|
||||
$bottom_x = $width;
|
||||
$bottom_y = $height;
|
||||
|
||||
// search upper part of image for colors to trim away
|
||||
if (in_array('top', $away)) {
|
||||
|
||||
for ($y=0; $y < ceil($height/2); $y++) {
|
||||
for ($x=0; $x < $width; $x++) {
|
||||
|
||||
$checkColor = $image->pickColor($x, $y, 'object');
|
||||
|
||||
if ($checkTransparency) {
|
||||
$checkColor->r = $color->r;
|
||||
$checkColor->g = $color->g;
|
||||
$checkColor->b = $color->b;
|
||||
}
|
||||
|
||||
if ($color->differs($checkColor, $tolerance)) {
|
||||
$top_y = max(0, $y - $feather);
|
||||
break 2;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// search left part of image for colors to trim away
|
||||
if (in_array('left', $away)) {
|
||||
|
||||
for ($x=0; $x < ceil($width/2); $x++) {
|
||||
for ($y=$top_y; $y < $height; $y++) {
|
||||
|
||||
$checkColor = $image->pickColor($x, $y, 'object');
|
||||
|
||||
if ($checkTransparency) {
|
||||
$checkColor->r = $color->r;
|
||||
$checkColor->g = $color->g;
|
||||
$checkColor->b = $color->b;
|
||||
}
|
||||
|
||||
if ($color->differs($checkColor, $tolerance)) {
|
||||
$top_x = max(0, $x - $feather);
|
||||
break 2;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// search lower part of image for colors to trim away
|
||||
if (in_array('bottom', $away)) {
|
||||
|
||||
for ($y=($height-1); $y >= floor($height/2)-1; $y--) {
|
||||
for ($x=$top_x; $x < $width; $x++) {
|
||||
|
||||
$checkColor = $image->pickColor($x, $y, 'object');
|
||||
|
||||
if ($checkTransparency) {
|
||||
$checkColor->r = $color->r;
|
||||
$checkColor->g = $color->g;
|
||||
$checkColor->b = $color->b;
|
||||
}
|
||||
|
||||
if ($color->differs($checkColor, $tolerance)) {
|
||||
$bottom_y = min($height, $y+1 + $feather);
|
||||
break 2;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// search right part of image for colors to trim away
|
||||
if (in_array('right', $away)) {
|
||||
|
||||
for ($x=($width-1); $x >= floor($width/2)-1; $x--) {
|
||||
for ($y=$top_y; $y < $bottom_y; $y++) {
|
||||
|
||||
$checkColor = $image->pickColor($x, $y, 'object');
|
||||
|
||||
if ($checkTransparency) {
|
||||
$checkColor->r = $color->r;
|
||||
$checkColor->g = $color->g;
|
||||
$checkColor->b = $color->b;
|
||||
}
|
||||
|
||||
if ($color->differs($checkColor, $tolerance)) {
|
||||
$bottom_x = min($width, $x+1 + $feather);
|
||||
break 2;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
// trim parts of image
|
||||
return $this->modify($image, 0, 0, $top_x, $top_y, ($bottom_x-$top_x), ($bottom_y-$top_y), ($bottom_x-$top_x), ($bottom_y-$top_y));
|
||||
|
||||
}
|
||||
}
|
28
vendor/intervention/image/src/Intervention/Image/Gd/Commands/WidenCommand.php
vendored
Normal file
28
vendor/intervention/image/src/Intervention/Image/Gd/Commands/WidenCommand.php
vendored
Normal file
@ -0,0 +1,28 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Commands;
|
||||
|
||||
class WidenCommand extends ResizeCommand
|
||||
{
|
||||
/**
|
||||
* Resize image proportionally to given width
|
||||
*
|
||||
* @param \Intervention\Image\Image $image
|
||||
* @return boolean
|
||||
*/
|
||||
public function execute($image)
|
||||
{
|
||||
$width = $this->argument(0)->type('digit')->required()->value();
|
||||
$additionalConstraints = $this->argument(1)->type('closure')->value();
|
||||
|
||||
$this->arguments[0] = $width;
|
||||
$this->arguments[1] = null;
|
||||
$this->arguments[2] = function ($constraint) use ($additionalConstraints) {
|
||||
$constraint->aspectRatio();
|
||||
if(is_callable($additionalConstraints))
|
||||
$additionalConstraints($constraint);
|
||||
};
|
||||
|
||||
return parent::execute($image);
|
||||
}
|
||||
}
|
171
vendor/intervention/image/src/Intervention/Image/Gd/Decoder.php
vendored
Normal file
171
vendor/intervention/image/src/Intervention/Image/Gd/Decoder.php
vendored
Normal file
@ -0,0 +1,171 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd;
|
||||
|
||||
use Intervention\Image\Exception\NotReadableException;
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class Decoder extends \Intervention\Image\AbstractDecoder
|
||||
{
|
||||
/**
|
||||
* Initiates new image from path in filesystem
|
||||
*
|
||||
* @param string $path
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromPath($path)
|
||||
{
|
||||
if ( ! file_exists($path)) {
|
||||
throw new NotReadableException(
|
||||
"Unable to find file ({$path})."
|
||||
);
|
||||
}
|
||||
|
||||
// get mime type of file
|
||||
$mime = finfo_file(finfo_open(FILEINFO_MIME_TYPE), $path);
|
||||
|
||||
// define core
|
||||
switch (strtolower($mime)) {
|
||||
case 'image/png':
|
||||
case 'image/x-png':
|
||||
$core = @imagecreatefrompng($path);
|
||||
break;
|
||||
|
||||
case 'image/jpg':
|
||||
case 'image/jpeg':
|
||||
case 'image/pjpeg':
|
||||
$core = @imagecreatefromjpeg($path);
|
||||
if (!$core) {
|
||||
$core= @imagecreatefromstring(file_get_contents($path));
|
||||
}
|
||||
break;
|
||||
|
||||
case 'image/gif':
|
||||
$core = @imagecreatefromgif($path);
|
||||
break;
|
||||
|
||||
case 'image/webp':
|
||||
case 'image/x-webp':
|
||||
if ( ! function_exists('imagecreatefromwebp')) {
|
||||
throw new NotReadableException(
|
||||
"Unsupported image type. GD/PHP installation does not support WebP format."
|
||||
);
|
||||
}
|
||||
$core = @imagecreatefromwebp($path);
|
||||
break;
|
||||
|
||||
case 'image/bmp':
|
||||
case 'image/ms-bmp':
|
||||
case 'image/x-bitmap':
|
||||
case 'image/x-bmp':
|
||||
case 'image/x-ms-bmp':
|
||||
case 'image/x-win-bitmap':
|
||||
case 'image/x-windows-bmp':
|
||||
case 'image/x-xbitmap':
|
||||
if (! function_exists('imagecreatefrombmp')) {
|
||||
throw new NotReadableException(
|
||||
"Unsupported image type. GD/PHP installation does not support BMP format."
|
||||
);
|
||||
}
|
||||
$core = @imagecreatefrombmp($path);
|
||||
break;
|
||||
|
||||
default:
|
||||
throw new NotReadableException(
|
||||
sprintf("Unsupported image type %s. GD driver is only able to decode JPG, PNG, GIF, BMP or WebP files.", strtolower($mime))
|
||||
);
|
||||
}
|
||||
|
||||
if (empty($core)) {
|
||||
throw new NotReadableException(
|
||||
"Unable to decode image from file ({$path})."
|
||||
);
|
||||
}
|
||||
|
||||
$this->gdResourceToTruecolor($core);
|
||||
|
||||
// build image
|
||||
$image = $this->initFromGdResource($core);
|
||||
$image->mime = $mime;
|
||||
$image->setFileInfoFromPath($path);
|
||||
|
||||
return $image;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates new image from GD resource
|
||||
*
|
||||
* @param Resource $resource
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromGdResource($resource)
|
||||
{
|
||||
return new Image(new Driver, $resource);
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates new image from Imagick object
|
||||
*
|
||||
* @param Imagick $object
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromImagick(\Imagick $object)
|
||||
{
|
||||
throw new NotSupportedException(
|
||||
"Gd driver is unable to init from Imagick object."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates new image from binary data
|
||||
*
|
||||
* @param string $data
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function initFromBinary($binary)
|
||||
{
|
||||
$resource = @imagecreatefromstring($binary);
|
||||
|
||||
if ($resource === false) {
|
||||
throw new NotReadableException(
|
||||
"Unable to init from given binary data."
|
||||
);
|
||||
}
|
||||
|
||||
$image = $this->initFromGdResource($resource);
|
||||
$image->mime = finfo_buffer(finfo_open(FILEINFO_MIME_TYPE), $binary);
|
||||
|
||||
return $image;
|
||||
}
|
||||
|
||||
/**
|
||||
* Transform GD resource into Truecolor version
|
||||
*
|
||||
* @param resource $resource
|
||||
* @return bool
|
||||
*/
|
||||
public function gdResourceToTruecolor(&$resource)
|
||||
{
|
||||
$width = imagesx($resource);
|
||||
$height = imagesy($resource);
|
||||
|
||||
// new canvas
|
||||
$canvas = imagecreatetruecolor($width, $height);
|
||||
|
||||
// fill with transparent color
|
||||
imagealphablending($canvas, false);
|
||||
$transparent = imagecolorallocatealpha($canvas, 255, 255, 255, 127);
|
||||
imagefilledrectangle($canvas, 0, 0, $width, $height, $transparent);
|
||||
imagecolortransparent($canvas, $transparent);
|
||||
imagealphablending($canvas, true);
|
||||
|
||||
// copy original
|
||||
imagecopy($canvas, $resource, 0, 0, 0, 0, $width, $height);
|
||||
imagedestroy($resource);
|
||||
|
||||
$resource = $canvas;
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
89
vendor/intervention/image/src/Intervention/Image/Gd/Driver.php
vendored
Normal file
89
vendor/intervention/image/src/Intervention/Image/Gd/Driver.php
vendored
Normal file
@ -0,0 +1,89 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd;
|
||||
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class Driver extends \Intervention\Image\AbstractDriver
|
||||
{
|
||||
/**
|
||||
* Creates new instance of driver
|
||||
*
|
||||
* @param Decoder $decoder
|
||||
* @param Encoder $encoder
|
||||
*/
|
||||
public function __construct(Decoder $decoder = null, Encoder $encoder = null)
|
||||
{
|
||||
if ( ! $this->coreAvailable()) {
|
||||
throw new NotSupportedException(
|
||||
"GD Library extension not available with this PHP installation."
|
||||
);
|
||||
}
|
||||
|
||||
$this->decoder = $decoder ? $decoder : new Decoder;
|
||||
$this->encoder = $encoder ? $encoder : new Encoder;
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates new image instance
|
||||
*
|
||||
* @param int $width
|
||||
* @param int $height
|
||||
* @param mixed $background
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function newImage($width, $height, $background = null)
|
||||
{
|
||||
// create empty resource
|
||||
$core = imagecreatetruecolor($width, $height);
|
||||
$image = new Image(new static, $core);
|
||||
|
||||
// set background color
|
||||
$background = new Color($background);
|
||||
imagefill($image->getCore(), 0, 0, $background->getInt());
|
||||
|
||||
return $image;
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads given string into color object
|
||||
*
|
||||
* @param string $value
|
||||
* @return AbstractColor
|
||||
*/
|
||||
public function parseColor($value)
|
||||
{
|
||||
return new Color($value);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if core module installation is available
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
protected function coreAvailable()
|
||||
{
|
||||
return (extension_loaded('gd') && function_exists('gd_info'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns clone of given core
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function cloneCore($core)
|
||||
{
|
||||
$width = imagesx($core);
|
||||
$height = imagesy($core);
|
||||
$clone = imagecreatetruecolor($width, $height);
|
||||
imagealphablending($clone, false);
|
||||
imagesavealpha($clone, true);
|
||||
$transparency = imagecolorallocatealpha($clone, 0, 0, 0, 127);
|
||||
imagefill($clone, 0, 0, $transparency);
|
||||
|
||||
imagecopy($clone, $core, 0, 0, 0, 0, $width, $height);
|
||||
|
||||
return $clone;
|
||||
}
|
||||
}
|
180
vendor/intervention/image/src/Intervention/Image/Gd/Encoder.php
vendored
Normal file
180
vendor/intervention/image/src/Intervention/Image/Gd/Encoder.php
vendored
Normal file
@ -0,0 +1,180 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd;
|
||||
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
class Encoder extends \Intervention\Image\AbstractEncoder
|
||||
{
|
||||
/**
|
||||
* Processes and returns encoded image as JPEG string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processJpeg()
|
||||
{
|
||||
ob_start();
|
||||
imagejpeg($this->image->getCore(), null, $this->quality);
|
||||
$this->image->mime = image_type_to_mime_type(IMAGETYPE_JPEG);
|
||||
$buffer = ob_get_contents();
|
||||
ob_end_clean();
|
||||
|
||||
return $buffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as PNG string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processPng()
|
||||
{
|
||||
ob_start();
|
||||
$resource = $this->image->getCore();
|
||||
imagealphablending($resource, false);
|
||||
imagesavealpha($resource, true);
|
||||
imagepng($resource, null, -1);
|
||||
$this->image->mime = image_type_to_mime_type(IMAGETYPE_PNG);
|
||||
$buffer = ob_get_contents();
|
||||
ob_end_clean();
|
||||
|
||||
return $buffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as GIF string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processGif()
|
||||
{
|
||||
ob_start();
|
||||
imagegif($this->image->getCore());
|
||||
$this->image->mime = image_type_to_mime_type(IMAGETYPE_GIF);
|
||||
$buffer = ob_get_contents();
|
||||
ob_end_clean();
|
||||
|
||||
return $buffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as WEBP string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processWebp()
|
||||
{
|
||||
if ( ! function_exists('imagewebp')) {
|
||||
throw new NotSupportedException(
|
||||
"Webp format is not supported by PHP installation."
|
||||
);
|
||||
}
|
||||
|
||||
ob_start();
|
||||
imagepalettetotruecolor($this->image->getCore());
|
||||
imagealphablending($this->image->getCore(), true);
|
||||
imagesavealpha($this->image->getCore(), true);
|
||||
imagewebp($this->image->getCore(), null, $this->quality);
|
||||
$this->image->mime = defined('IMAGETYPE_WEBP') ? image_type_to_mime_type(IMAGETYPE_WEBP) : 'image/webp';
|
||||
$buffer = ob_get_contents();
|
||||
ob_end_clean();
|
||||
|
||||
return $buffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as TIFF string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processTiff()
|
||||
{
|
||||
throw new NotSupportedException(
|
||||
"TIFF format is not supported by Gd Driver."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as BMP string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processBmp()
|
||||
{
|
||||
if ( ! function_exists('imagebmp')) {
|
||||
throw new NotSupportedException(
|
||||
"BMP format is not supported by PHP installation."
|
||||
);
|
||||
}
|
||||
|
||||
ob_start();
|
||||
imagebmp($this->image->getCore());
|
||||
$this->image->mime = defined('IMAGETYPE_BMP') ? image_type_to_mime_type(IMAGETYPE_BMP) : 'image/bmp';
|
||||
$buffer = ob_get_contents();
|
||||
ob_end_clean();
|
||||
|
||||
return $buffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as ICO string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processIco()
|
||||
{
|
||||
throw new NotSupportedException(
|
||||
"ICO format is not supported by Gd Driver."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as PSD string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processPsd()
|
||||
{
|
||||
throw new NotSupportedException(
|
||||
"PSD format is not supported by Gd Driver."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as AVIF string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processAvif()
|
||||
{
|
||||
if ( ! function_exists('imageavif')) {
|
||||
throw new NotSupportedException(
|
||||
"AVIF format is not supported by PHP installation."
|
||||
);
|
||||
}
|
||||
|
||||
ob_start();
|
||||
$resource = $this->image->getCore();
|
||||
imagepalettetotruecolor($resource);
|
||||
imagealphablending($resource, true);
|
||||
imagesavealpha($resource, true);
|
||||
imageavif($resource, null, $this->quality);
|
||||
$this->image->mime = defined('IMAGETYPE_AVIF') ? image_type_to_mime_type(IMAGETYPE_AVIF) : 'image/avif';
|
||||
$buffer = ob_get_contents();
|
||||
ob_end_clean();
|
||||
|
||||
return $buffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Processes and returns encoded image as HEIC string
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function processHeic()
|
||||
{
|
||||
throw new NotSupportedException(
|
||||
"HEIC format is not supported by Gd Driver."
|
||||
);
|
||||
}
|
||||
}
|
277
vendor/intervention/image/src/Intervention/Image/Gd/Font.php
vendored
Normal file
277
vendor/intervention/image/src/Intervention/Image/Gd/Font.php
vendored
Normal file
@ -0,0 +1,277 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd;
|
||||
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class Font extends \Intervention\Image\AbstractFont
|
||||
{
|
||||
/**
|
||||
* Get font size in points
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
protected function getPointSize()
|
||||
{
|
||||
return intval(ceil($this->size * 0.75));
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter function to access internal integer font values
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
private function getInternalFont()
|
||||
{
|
||||
$internalfont = is_null($this->file) ? 1 : $this->file;
|
||||
$internalfont = is_numeric($internalfont) ? $internalfont : false;
|
||||
|
||||
if ( ! in_array($internalfont, [1, 2, 3, 4, 5])) {
|
||||
throw new NotSupportedException(
|
||||
sprintf('Internal GD font (%s) not available. Use only 1-5.', $internalfont)
|
||||
);
|
||||
}
|
||||
|
||||
return intval($internalfont);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get width of an internal font character
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
private function getInternalFontWidth()
|
||||
{
|
||||
return $this->getInternalFont() + 4;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get height of an internal font character
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
private function getInternalFontHeight()
|
||||
{
|
||||
switch ($this->getInternalFont()) {
|
||||
case 1:
|
||||
return 8;
|
||||
|
||||
case 2:
|
||||
return 14;
|
||||
|
||||
case 3:
|
||||
return 14;
|
||||
|
||||
case 4:
|
||||
return 16;
|
||||
|
||||
case 5:
|
||||
return 16;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates bounding box of current font setting
|
||||
*
|
||||
* @return Array
|
||||
*/
|
||||
public function getBoxSize()
|
||||
{
|
||||
$box = [];
|
||||
|
||||
if ($this->hasApplicableFontFile()) {
|
||||
|
||||
// imagettfbbox() converts numeric entities to their respective
|
||||
// character. Preserve any originally double encoded entities to be
|
||||
// represented as is.
|
||||
// eg: &#160; will render   rather than its character.
|
||||
$this->text = preg_replace('/&(#(?:x[a-fA-F0-9]+|[0-9]+);)/', '&\1', $this->text);
|
||||
$this->text = mb_encode_numericentity($this->text, array(0x0080, 0xffff, 0, 0xffff), 'UTF-8');
|
||||
|
||||
// get bounding box with angle 0
|
||||
$box = imagettfbbox($this->getPointSize(), 0, $this->file, $this->text);
|
||||
|
||||
// rotate points manually
|
||||
if ($this->angle != 0) {
|
||||
|
||||
$angle = pi() * 2 - $this->angle * pi() * 2 / 360;
|
||||
|
||||
for ($i=0; $i<4; $i++) {
|
||||
$x = $box[$i * 2];
|
||||
$y = $box[$i * 2 + 1];
|
||||
$box[$i * 2] = cos($angle) * $x - sin($angle) * $y;
|
||||
$box[$i * 2 + 1] = sin($angle) * $x + cos($angle) * $y;
|
||||
}
|
||||
}
|
||||
|
||||
$box['width'] = intval(abs($box[4] - $box[0]));
|
||||
$box['height'] = intval(abs($box[5] - $box[1]));
|
||||
|
||||
} else {
|
||||
|
||||
// get current internal font size
|
||||
$width = $this->getInternalFontWidth();
|
||||
$height = $this->getInternalFontHeight();
|
||||
|
||||
if (strlen($this->text) == 0) {
|
||||
// no text -> no boxsize
|
||||
$box['width'] = 0;
|
||||
$box['height'] = 0;
|
||||
} else {
|
||||
// calculate boxsize
|
||||
$box['width'] = strlen($this->text) * $width;
|
||||
$box['height'] = $height;
|
||||
}
|
||||
}
|
||||
|
||||
return $box;
|
||||
}
|
||||
|
||||
/**
|
||||
* Draws font to given image at given position
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $posx
|
||||
* @param int $posy
|
||||
* @return void
|
||||
*/
|
||||
public function applyToImage(Image $image, $posx = 0, $posy = 0)
|
||||
{
|
||||
// parse text color
|
||||
$color = new Color($this->color);
|
||||
|
||||
if ($this->hasApplicableFontFile()) {
|
||||
|
||||
if ($this->angle != 0 || is_string($this->align) || is_string($this->valign)) {
|
||||
|
||||
$box = $this->getBoxSize();
|
||||
|
||||
$align = is_null($this->align) ? 'left' : strtolower($this->align);
|
||||
$valign = is_null($this->valign) ? 'bottom' : strtolower($this->valign);
|
||||
|
||||
// correction on position depending on v/h alignment
|
||||
switch ($align.'-'.$valign) {
|
||||
|
||||
case 'center-top':
|
||||
$posx = $posx - round(($box[6]+$box[4])/2);
|
||||
$posy = $posy - round(($box[7]+$box[5])/2);
|
||||
break;
|
||||
|
||||
case 'right-top':
|
||||
$posx = $posx - $box[4];
|
||||
$posy = $posy - $box[5];
|
||||
break;
|
||||
|
||||
case 'left-top':
|
||||
$posx = $posx - $box[6];
|
||||
$posy = $posy - $box[7];
|
||||
break;
|
||||
|
||||
case 'center-center':
|
||||
case 'center-middle':
|
||||
$posx = $posx - round(($box[0]+$box[4])/2);
|
||||
$posy = $posy - round(($box[1]+$box[5])/2);
|
||||
break;
|
||||
|
||||
case 'right-center':
|
||||
case 'right-middle':
|
||||
$posx = $posx - round(($box[2]+$box[4])/2);
|
||||
$posy = $posy - round(($box[3]+$box[5])/2);
|
||||
break;
|
||||
|
||||
case 'left-center':
|
||||
case 'left-middle':
|
||||
$posx = $posx - round(($box[0]+$box[6])/2);
|
||||
$posy = $posy - round(($box[1]+$box[7])/2);
|
||||
break;
|
||||
|
||||
case 'center-bottom':
|
||||
$posx = $posx - round(($box[0]+$box[2])/2);
|
||||
$posy = $posy - round(($box[1]+$box[3])/2);
|
||||
break;
|
||||
|
||||
case 'right-bottom':
|
||||
$posx = $posx - $box[2];
|
||||
$posy = $posy - $box[3];
|
||||
break;
|
||||
|
||||
case 'left-bottom':
|
||||
$posx = $posx - $box[0];
|
||||
$posy = $posy - $box[1];
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// enable alphablending for imagettftext
|
||||
imagealphablending($image->getCore(), true);
|
||||
|
||||
// draw ttf text
|
||||
imagettftext($image->getCore(), $this->getPointSize(), $this->angle, $posx, $posy, $color->getInt(), $this->file, $this->text);
|
||||
|
||||
} else {
|
||||
|
||||
// get box size
|
||||
$box = $this->getBoxSize();
|
||||
$width = $box['width'];
|
||||
$height = $box['height'];
|
||||
|
||||
// internal font specific position corrections
|
||||
if ($this->getInternalFont() == 1) {
|
||||
$top_correction = 1;
|
||||
$bottom_correction = 2;
|
||||
} elseif ($this->getInternalFont() == 3) {
|
||||
$top_correction = 2;
|
||||
$bottom_correction = 4;
|
||||
} else {
|
||||
$top_correction = 3;
|
||||
$bottom_correction = 4;
|
||||
}
|
||||
|
||||
// x-position corrections for horizontal alignment
|
||||
switch (strtolower($this->align)) {
|
||||
case 'center':
|
||||
$posx = ceil($posx - ($width / 2));
|
||||
break;
|
||||
|
||||
case 'right':
|
||||
$posx = ceil($posx - $width) + 1;
|
||||
break;
|
||||
}
|
||||
|
||||
// y-position corrections for vertical alignment
|
||||
switch (strtolower($this->valign)) {
|
||||
case 'center':
|
||||
case 'middle':
|
||||
$posy = ceil($posy - ($height / 2));
|
||||
break;
|
||||
|
||||
case 'top':
|
||||
$posy = ceil($posy - $top_correction);
|
||||
break;
|
||||
|
||||
default:
|
||||
case 'bottom':
|
||||
$posy = round($posy - $height + $bottom_correction);
|
||||
break;
|
||||
}
|
||||
|
||||
// draw text
|
||||
imagestring($image->getCore(), $this->getInternalFont(), $posx, $posy, $this->text, $color->getInt());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Set text kerning
|
||||
*
|
||||
* @param string $kerning
|
||||
* @return void
|
||||
*/
|
||||
public function kerning($kerning)
|
||||
{
|
||||
throw new \Intervention\Image\Exception\NotSupportedException(
|
||||
"Kerning is not supported by GD driver."
|
||||
);
|
||||
}
|
||||
|
||||
}
|
40
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/CircleShape.php
vendored
Normal file
40
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/CircleShape.php
vendored
Normal file
@ -0,0 +1,40 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Shapes;
|
||||
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class CircleShape extends EllipseShape
|
||||
{
|
||||
/**
|
||||
* Diameter of circle in pixels
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $diameter = 100;
|
||||
|
||||
/**
|
||||
* Create new instance of circle
|
||||
*
|
||||
* @param int $diameter
|
||||
*/
|
||||
public function __construct($diameter = null)
|
||||
{
|
||||
$this->width = is_numeric($diameter) ? intval($diameter) : $this->diameter;
|
||||
$this->height = is_numeric($diameter) ? intval($diameter) : $this->diameter;
|
||||
$this->diameter = is_numeric($diameter) ? intval($diameter) : $this->diameter;
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw current circle on given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $x
|
||||
* @param int $y
|
||||
* @return boolean
|
||||
*/
|
||||
public function applyToImage(Image $image, $x = 0, $y = 0)
|
||||
{
|
||||
return parent::applyToImage($image, $x, $y);
|
||||
}
|
||||
}
|
65
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/EllipseShape.php
vendored
Normal file
65
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/EllipseShape.php
vendored
Normal file
@ -0,0 +1,65 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Shapes;
|
||||
|
||||
use Intervention\Image\AbstractShape;
|
||||
use Intervention\Image\Gd\Color;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class EllipseShape extends AbstractShape
|
||||
{
|
||||
/**
|
||||
* Width of ellipse in pixels
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $width = 100;
|
||||
|
||||
/**
|
||||
* Height of ellipse in pixels
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $height = 100;
|
||||
|
||||
/**
|
||||
* Create new ellipse instance
|
||||
*
|
||||
* @param int $width
|
||||
* @param int $height
|
||||
*/
|
||||
public function __construct($width = null, $height = null)
|
||||
{
|
||||
$this->width = is_numeric($width) ? intval($width) : $this->width;
|
||||
$this->height = is_numeric($height) ? intval($height) : $this->height;
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw ellipse instance on given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $x
|
||||
* @param int $y
|
||||
* @return boolean
|
||||
*/
|
||||
public function applyToImage(Image $image, $x = 0, $y = 0)
|
||||
{
|
||||
// parse background color
|
||||
$background = new Color($this->background);
|
||||
|
||||
if ($this->hasBorder()) {
|
||||
// slightly smaller ellipse to keep 1px bordered edges clean
|
||||
imagefilledellipse($image->getCore(), $x, $y, $this->width-1, $this->height-1, $background->getInt());
|
||||
|
||||
$border_color = new Color($this->border_color);
|
||||
imagesetthickness($image->getCore(), $this->border_width);
|
||||
|
||||
// gd's imageellipse doesn't respect imagesetthickness so i use imagearc with 359.9 degrees here
|
||||
imagearc($image->getCore(), $x, $y, $this->width, $this->height, 0, 359.99, $border_color->getInt());
|
||||
} else {
|
||||
imagefilledellipse($image->getCore(), $x, $y, $this->width, $this->height, $background->getInt());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
90
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/LineShape.php
vendored
Normal file
90
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/LineShape.php
vendored
Normal file
@ -0,0 +1,90 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Shapes;
|
||||
|
||||
use Intervention\Image\AbstractShape;
|
||||
use Intervention\Image\Gd\Color;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class LineShape extends AbstractShape
|
||||
{
|
||||
/**
|
||||
* Starting point x-coordinate of line
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $x = 0;
|
||||
|
||||
/**
|
||||
* Starting point y-coordinate of line
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $y = 0;
|
||||
|
||||
/**
|
||||
* Color of line
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $color = '#000000';
|
||||
|
||||
/**
|
||||
* Width of line in pixels
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $width = 1;
|
||||
|
||||
/**
|
||||
* Create new line shape instance
|
||||
*
|
||||
* @param int $x
|
||||
* @param int $y
|
||||
*/
|
||||
public function __construct($x = null, $y = null)
|
||||
{
|
||||
$this->x = is_numeric($x) ? intval($x) : $this->x;
|
||||
$this->y = is_numeric($y) ? intval($y) : $this->y;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set current line color
|
||||
*
|
||||
* @param string $color
|
||||
* @return void
|
||||
*/
|
||||
public function color($color)
|
||||
{
|
||||
$this->color = $color;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set current line width in pixels
|
||||
*
|
||||
* @param int $width
|
||||
* @return void
|
||||
*/
|
||||
public function width($width)
|
||||
{
|
||||
throw new \Intervention\Image\Exception\NotSupportedException(
|
||||
"Line width is not supported by GD driver."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw current instance of line to given endpoint on given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $x
|
||||
* @param int $y
|
||||
* @return boolean
|
||||
*/
|
||||
public function applyToImage(Image $image, $x = 0, $y = 0)
|
||||
{
|
||||
$color = new Color($this->color);
|
||||
imageline($image->getCore(), $x, $y, $this->x, $this->y, $color->getInt());
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
49
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/PolygonShape.php
vendored
Normal file
49
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/PolygonShape.php
vendored
Normal file
@ -0,0 +1,49 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Shapes;
|
||||
|
||||
use Intervention\Image\AbstractShape;
|
||||
use Intervention\Image\Gd\Color;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class PolygonShape extends AbstractShape
|
||||
{
|
||||
/**
|
||||
* Array of points of polygon
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $points;
|
||||
|
||||
/**
|
||||
* Create new polygon instance
|
||||
*
|
||||
* @param array $points
|
||||
*/
|
||||
public function __construct($points)
|
||||
{
|
||||
$this->points = $points;
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw polygon on given image
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $x
|
||||
* @param int $y
|
||||
* @return boolean
|
||||
*/
|
||||
public function applyToImage(Image $image, $x = 0, $y = 0)
|
||||
{
|
||||
$background = new Color($this->background);
|
||||
imagefilledpolygon($image->getCore(), $this->points, intval(count($this->points) / 2), $background->getInt());
|
||||
|
||||
if ($this->hasBorder()) {
|
||||
$border_color = new Color($this->border_color);
|
||||
imagesetthickness($image->getCore(), $this->border_width);
|
||||
imagepolygon($image->getCore(), $this->points, intval(count($this->points) / 2), $border_color->getInt());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
76
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/RectangleShape.php
vendored
Normal file
76
vendor/intervention/image/src/Intervention/Image/Gd/Shapes/RectangleShape.php
vendored
Normal file
@ -0,0 +1,76 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image\Gd\Shapes;
|
||||
|
||||
use Intervention\Image\AbstractShape;
|
||||
use Intervention\Image\Gd\Color;
|
||||
use Intervention\Image\Image;
|
||||
|
||||
class RectangleShape extends AbstractShape
|
||||
{
|
||||
/**
|
||||
* X-Coordinate of top-left point
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $x1 = 0;
|
||||
|
||||
/**
|
||||
* Y-Coordinate of top-left point
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $y1 = 0;
|
||||
|
||||
/**
|
||||
* X-Coordinate of bottom-right point
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $x2 = 0;
|
||||
|
||||
/**
|
||||
* Y-Coordinate of bottom-right point
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
public $y2 = 0;
|
||||
|
||||
/**
|
||||
* Create new rectangle shape instance
|
||||
*
|
||||
* @param int $x1
|
||||
* @param int $y1
|
||||
* @param int $x2
|
||||
* @param int $y2
|
||||
*/
|
||||
public function __construct($x1 = null, $y1 = null, $x2 = null, $y2 = null)
|
||||
{
|
||||
$this->x1 = is_numeric($x1) ? intval($x1) : $this->x1;
|
||||
$this->y1 = is_numeric($y1) ? intval($y1) : $this->y1;
|
||||
$this->x2 = is_numeric($x2) ? intval($x2) : $this->x2;
|
||||
$this->y2 = is_numeric($y2) ? intval($y2) : $this->y2;
|
||||
}
|
||||
|
||||
/**
|
||||
* Draw rectangle to given image at certain position
|
||||
*
|
||||
* @param Image $image
|
||||
* @param int $x
|
||||
* @param int $y
|
||||
* @return boolean
|
||||
*/
|
||||
public function applyToImage(Image $image, $x = 0, $y = 0)
|
||||
{
|
||||
$background = new Color($this->background);
|
||||
imagefilledrectangle($image->getCore(), $this->x1, $this->y1, $this->x2, $this->y2, $background->getInt());
|
||||
|
||||
if ($this->hasBorder()) {
|
||||
$border_color = new Color($this->border_color);
|
||||
imagesetthickness($image->getCore(), $this->border_width);
|
||||
imagerectangle($image->getCore(), $this->x1, $this->y1, $this->x2, $this->y2, $border_color->getInt());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
370
vendor/intervention/image/src/Intervention/Image/Image.php
vendored
Normal file
370
vendor/intervention/image/src/Intervention/Image/Image.php
vendored
Normal file
@ -0,0 +1,370 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Intervention\Image\Exception\NotWritableException;
|
||||
use Intervention\Image\Exception\RuntimeException;
|
||||
use Psr\Http\Message\ResponseInterface;
|
||||
use Psr\Http\Message\StreamInterface;
|
||||
|
||||
/**
|
||||
* @method \Intervention\Image\Image backup(string $name = 'default') Backups current image state as fallback for reset method under an optional name. Overwrites older state on every call, unless a different name is passed.
|
||||
* @method \Intervention\Image\Image blur(int $amount = 1) Apply a gaussian blur filter with a optional amount on the current image. Use values between 0 and 100.
|
||||
* @method \Intervention\Image\Image brightness(int $level) Changes the brightness of the current image by the given level. Use values between -100 for min. brightness. 0 for no change and +100 for max. brightness.
|
||||
* @method \Intervention\Image\Image cache(\Closure $callback, int $lifetime = null, boolean $returnObj = false) Method to create a new cached image instance from a Closure callback. Pass a lifetime in minutes for the callback and decide whether you want to get an Intervention Image instance as return value or just receive the image stream.
|
||||
* @method \Intervention\Image\Image canvas(int $width, int $height, mixed $bgcolor = null) Factory method to create a new empty image instance with given width and height. You can define a background-color optionally. By default the canvas background is transparent.
|
||||
* @method \Intervention\Image\Image circle(int $diameter, int $x, int $y, \Closure $callback = null) Draw a circle at given x, y, coordinates with given diameter. You can define the appearance of the circle by an optional closure callback.
|
||||
* @method \Intervention\Image\Image colorize(int $red, int $green, int $blue) Change the RGB color values of the current image on the given channels red, green and blue. The input values are normalized so you have to include parameters from 100 for maximum color value. 0 for no change and -100 to take out all the certain color on the image.
|
||||
* @method \Intervention\Image\Image contrast(int $level) Changes the contrast of the current image by the given level. Use values between -100 for min. contrast 0 for no change and +100 for max. contrast.
|
||||
* @method \Intervention\Image\Image crop(int $width, int $height, int $x = null, int $y = null) Cut out a rectangular part of the current image with given width and height. Define optional x,y coordinates to move the top-left corner of the cutout to a certain position.
|
||||
* @method void destroy() Frees memory associated with the current image instance before the PHP script ends. Normally resources are destroyed automatically after the script is finished.
|
||||
* @method \Intervention\Image\Image ellipse(int $width, int $height, int $x, int $y, \Closure $callback = null) Draw a colored ellipse at given x, y, coordinates. You can define width and height and set the appearance of the circle by an optional closure callback.
|
||||
* @method mixed exif(string $key = null) Read Exif meta data from current image.
|
||||
* @method mixed iptc(string $key = null) Read Iptc meta data from current image.
|
||||
* @method \Intervention\Image\Image fill(mixed $filling, int $x = null, int $y = null) Fill current image with given color or another image used as tile for filling. Pass optional x, y coordinates to start at a certain point.
|
||||
* @method \Intervention\Image\Image flip(string $mode = 'h') Mirror the current image horizontally or vertically by specifying the mode.
|
||||
* @method \Intervention\Image\Image fit(int $width, int $height = null, \Closure $callback = null, string $position = 'center') Combine cropping and resizing to format image in a smart way. The method will find the best fitting aspect ratio of your given width and height on the current image automatically, cut it out and resize it to the given dimension. You may pass an optional Closure callback as third parameter, to prevent possible upsizing and a custom position of the cutout as fourth parameter.
|
||||
* @method \Intervention\Image\Image gamma(float $correction) Performs a gamma correction operation on the current image.
|
||||
* @method \Intervention\Image\Image greyscale() Turns image into a greyscale version.
|
||||
* @method \Intervention\Image\Image heighten(int $height, \Closure $callback = null) Resizes the current image to new height, constraining aspect ratio. Pass an optional Closure callback as third parameter, to apply additional constraints like preventing possible upsizing.
|
||||
* @method \Intervention\Image\Image insert(mixed $source, string $position = 'top-left', int $x = 0, int $y = 0) Paste a given image source over the current image with an optional position and a offset coordinate. This method can be used to apply another image as watermark because the transparency values are maintained.
|
||||
* @method \Intervention\Image\Image interlace(boolean $interlace = true) Determine whether an image should be encoded in interlaced or standard mode by toggling interlace mode with a boolean parameter. If an JPEG image is set interlaced the image will be processed as a progressive JPEG.
|
||||
* @method \Intervention\Image\Image invert() Reverses all colors of the current image.
|
||||
* @method \Intervention\Image\Image limitColors(int $count, mixed $matte = null) Method converts the existing colors of the current image into a color table with a given maximum count of colors. The function preserves as much alpha channel information as possible and blends transarent pixels against a optional matte color.
|
||||
* @method \Intervention\Image\Image line(int $x1, int $y1, int $x2, int $y2, \Closure $callback = null) Draw a line from x,y point 1 to x,y point 2 on current image. Define color and/or width of line in an optional Closure callback.
|
||||
* @method \Intervention\Image\Image make(mixed $source) Universal factory method to create a new image instance from source, which can be a filepath, a GD image resource, an Imagick object or a binary image data.
|
||||
* @method \Intervention\Image\Image mask(mixed $source, boolean $mask_with_alpha) Apply a given image source as alpha mask to the current image to change current opacity. Mask will be resized to the current image size. By default a greyscale version of the mask is converted to alpha values, but you can set mask_with_alpha to apply the actual alpha channel. Any transparency values of the current image will be maintained.
|
||||
* @method \Intervention\Image\Image opacity(int $transparency) Set the opacity in percent of the current image ranging from 100% for opaque and 0% for full transparency.
|
||||
* @method \Intervention\Image\Image orientate() This method reads the EXIF image profile setting 'Orientation' and performs a rotation on the image to display the image correctly.
|
||||
* @method mixed pickColor(int $x, int $y, string $format = 'array') Pick a color at point x, y out of current image and return in optional given format.
|
||||
* @method \Intervention\Image\Image pixel(mixed $color, int $x, int $y) Draw a single pixel in given color on x, y position.
|
||||
* @method \Intervention\Image\Image pixelate(int $size) Applies a pixelation effect to the current image with a given size of pixels.
|
||||
* @method \Intervention\Image\Image polygon(array $points, \Closure $callback = null) Draw a colored polygon with given points. You can define the appearance of the polygon by an optional closure callback.
|
||||
* @method \Intervention\Image\Image rectangle(int $x1, int $y1, int $x2, int $y2, \Closure $callback = null) Draw a colored rectangle on current image with top-left corner on x,y point 1 and bottom-right corner at x,y point 2. Define the overall appearance of the shape by passing a Closure callback as an optional parameter.
|
||||
* @method \Intervention\Image\Image reset(string $name = 'default') Resets all of the modifications to a state saved previously by backup under an optional name.
|
||||
* @method \Intervention\Image\Image resize(int $width = null, int $height = null, \Closure $callback = null) Resizes current image based on given width and/or height. To contraint the resize command, pass an optional Closure callback as third parameter.
|
||||
* @method \Intervention\Image\Image resizeCanvas(int $width, int $height, string $anchor = 'center', boolean $relative = false, mixed $bgcolor = null) Resize the boundaries of the current image to given width and height. An anchor can be defined to determine from what point of the image the resizing is going to happen. Set the mode to relative to add or subtract the given width or height to the actual image dimensions. You can also pass a background color for the emerging area of the image.
|
||||
* @method mixed response(string $format = null, int $quality = 90) Sends HTTP response with current image in given format and quality.
|
||||
* @method \Intervention\Image\Image rotate(float $angle, mixed $bgcolor = null) Rotate the current image counter-clockwise by a given angle. Optionally define a background color for the uncovered zone after the rotation.
|
||||
* @method \Intervention\Image\Image sharpen(int $amount = 10) Sharpen current image with an optional amount. Use values between 0 and 100.
|
||||
* @method \Intervention\Image\Image text(string $text, int $x = 0, int $y = 0, \Closure $callback = null) Write a text string to the current image at an optional x,y basepoint position. You can define more details like font-size, font-file and alignment via a callback as the fourth parameter.
|
||||
* @method \Intervention\Image\Image trim(string $base = 'top-left', array $away = array('top', 'bottom', 'left', 'right'), int $tolerance = 0, int $feather = 0) Trim away image space in given color. Define an optional base to pick a color at a certain position and borders that should be trimmed away. You can also set an optional tolerance level, to trim similar colors and add a feathering border around the trimed image.
|
||||
* @method \Intervention\Image\Image widen(int $width, \Closure $callback = null) Resizes the current image to new width, constraining aspect ratio. Pass an optional Closure callback as third parameter, to apply additional constraints like preventing possible upsizing.
|
||||
* @method StreamInterface stream(string $format = null, int $quality = 90) Build PSR-7 compatible StreamInterface with current image in given format and quality.
|
||||
* @method ResponseInterface psrResponse(string $format = null, int $quality = 90) Build PSR-7 compatible ResponseInterface with current image in given format and quality.
|
||||
*/
|
||||
class Image extends File
|
||||
{
|
||||
/**
|
||||
* Instance of current image driver
|
||||
*
|
||||
* @var AbstractDriver
|
||||
*/
|
||||
protected $driver;
|
||||
|
||||
/**
|
||||
* Image resource/object of current image processor
|
||||
*
|
||||
* @var mixed
|
||||
*/
|
||||
protected $core;
|
||||
|
||||
/**
|
||||
* Array of Image resource backups of current image processor
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $backups = [];
|
||||
|
||||
/**
|
||||
* Last image encoding result
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $encoded = '';
|
||||
|
||||
/**
|
||||
* Creates a new Image instance
|
||||
*
|
||||
* @param AbstractDriver $driver
|
||||
* @param mixed $core
|
||||
*/
|
||||
public function __construct(AbstractDriver $driver = null, $core = null)
|
||||
{
|
||||
$this->driver = $driver;
|
||||
$this->core = $core;
|
||||
}
|
||||
|
||||
/**
|
||||
* Magic method to catch all image calls
|
||||
* usually any AbstractCommand
|
||||
*
|
||||
* @param string $name
|
||||
* @param Array $arguments
|
||||
* @return mixed
|
||||
*/
|
||||
public function __call($name, $arguments)
|
||||
{
|
||||
$command = $this->driver->executeCommand($this, $name, $arguments);
|
||||
return $command->hasOutput() ? $command->getOutput() : $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Starts encoding of current image
|
||||
*
|
||||
* @param string $format
|
||||
* @param int $quality
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function encode($format = null, $quality = 90)
|
||||
{
|
||||
return $this->driver->encode($this, $format, $quality);
|
||||
}
|
||||
|
||||
/**
|
||||
* Saves encoded image in filesystem
|
||||
*
|
||||
* @param string $path
|
||||
* @param int $quality
|
||||
* @param string $format
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function save($path = null, $quality = null, $format = null)
|
||||
{
|
||||
$path = is_null($path) ? $this->basePath() : $path;
|
||||
|
||||
if (is_null($path)) {
|
||||
throw new NotWritableException(
|
||||
"Can't write to undefined path."
|
||||
);
|
||||
}
|
||||
|
||||
if ($format === null) {
|
||||
$format = pathinfo($path, PATHINFO_EXTENSION);
|
||||
}
|
||||
|
||||
$data = $this->encode($format, $quality);
|
||||
$saved = @file_put_contents($path, $data);
|
||||
|
||||
if ($saved === false) {
|
||||
throw new NotWritableException(
|
||||
"Can't write image data to path ({$path})"
|
||||
);
|
||||
}
|
||||
|
||||
// set new file info
|
||||
$this->setFileInfoFromPath($path);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Runs a given filter on current image
|
||||
*
|
||||
* @param FiltersFilterInterface $filter
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function filter(Filters\FilterInterface $filter)
|
||||
{
|
||||
return $filter->applyFilter($this);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns current image driver
|
||||
*
|
||||
* @return \Intervention\Image\AbstractDriver
|
||||
*/
|
||||
public function getDriver()
|
||||
{
|
||||
return $this->driver;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets current image driver
|
||||
* @param AbstractDriver $driver
|
||||
*/
|
||||
public function setDriver(AbstractDriver $driver)
|
||||
{
|
||||
$this->driver = $driver;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns current image resource/obj
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public function getCore()
|
||||
{
|
||||
return $this->core;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets current image resource
|
||||
*
|
||||
* @param mixed $core
|
||||
*/
|
||||
public function setCore($core)
|
||||
{
|
||||
$this->core = $core;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns current image backup
|
||||
*
|
||||
* @param string $name
|
||||
* @return mixed
|
||||
*/
|
||||
public function getBackup($name = null)
|
||||
{
|
||||
$name = is_null($name) ? 'default' : $name;
|
||||
|
||||
if ( ! $this->backupExists($name)) {
|
||||
throw new RuntimeException(
|
||||
"Backup with name ({$name}) not available. Call backup() before reset()."
|
||||
);
|
||||
}
|
||||
|
||||
return $this->backups[$name];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns all backups attached to image
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function getBackups()
|
||||
{
|
||||
return $this->backups;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets current image backup
|
||||
*
|
||||
* @param mixed $resource
|
||||
* @param string $name
|
||||
* @return self
|
||||
*/
|
||||
public function setBackup($resource, $name = null)
|
||||
{
|
||||
$name = is_null($name) ? 'default' : $name;
|
||||
|
||||
$this->backups[$name] = $resource;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if named backup exists
|
||||
*
|
||||
* @param string $name
|
||||
* @return bool
|
||||
*/
|
||||
private function backupExists($name)
|
||||
{
|
||||
return array_key_exists($name, $this->backups);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if current image is already encoded
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
public function isEncoded()
|
||||
{
|
||||
return ! empty($this->encoded);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns encoded image data of current image
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function getEncoded()
|
||||
{
|
||||
return $this->encoded;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets encoded image buffer
|
||||
*
|
||||
* @param string $value
|
||||
*/
|
||||
public function setEncoded($value)
|
||||
{
|
||||
$this->encoded = $value;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates current image width
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getWidth()
|
||||
{
|
||||
return $this->getSize()->width;
|
||||
}
|
||||
|
||||
/**
|
||||
* Alias of getWidth()
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function width()
|
||||
{
|
||||
return $this->getWidth();
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates current image height
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function getHeight()
|
||||
{
|
||||
return $this->getSize()->height;
|
||||
}
|
||||
|
||||
/**
|
||||
* Alias of getHeight
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
public function height()
|
||||
{
|
||||
return $this->getHeight();
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads mime type
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function mime()
|
||||
{
|
||||
return $this->mime;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns encoded image data in string conversion
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function __toString()
|
||||
{
|
||||
return $this->encoded;
|
||||
}
|
||||
|
||||
/**
|
||||
* Cloning an image
|
||||
*/
|
||||
public function __clone()
|
||||
{
|
||||
$this->core = $this->driver->cloneCore($this->core);
|
||||
}
|
||||
}
|
142
vendor/intervention/image/src/Intervention/Image/ImageManager.php
vendored
Normal file
142
vendor/intervention/image/src/Intervention/Image/ImageManager.php
vendored
Normal file
@ -0,0 +1,142 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Closure;
|
||||
use Intervention\Image\Exception\MissingDependencyException;
|
||||
use Intervention\Image\Exception\NotSupportedException;
|
||||
|
||||
class ImageManager
|
||||
{
|
||||
/**
|
||||
* Config
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
public $config = [
|
||||
'driver' => 'gd'
|
||||
];
|
||||
|
||||
/**
|
||||
* Creates new instance of Image Manager
|
||||
*
|
||||
* @param array $config
|
||||
*/
|
||||
public function __construct(array $config = [])
|
||||
{
|
||||
$this->checkRequirements();
|
||||
$this->configure($config);
|
||||
}
|
||||
|
||||
/**
|
||||
* Overrides configuration settings
|
||||
*
|
||||
* @param array $config
|
||||
*
|
||||
* @return self
|
||||
*/
|
||||
public function configure(array $config = [])
|
||||
{
|
||||
$this->config = array_replace($this->config, $config);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initiates an Image instance from different input types
|
||||
*
|
||||
* @param mixed $data
|
||||
*
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function make($data)
|
||||
{
|
||||
return $this->createDriver()->init($data);
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates an empty image canvas
|
||||
*
|
||||
* @param int $width
|
||||
* @param int $height
|
||||
* @param mixed $background
|
||||
*
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public function canvas($width, $height, $background = null)
|
||||
{
|
||||
return $this->createDriver()->newImage($width, $height, $background);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create new cached image and run callback
|
||||
* (requires additional package intervention/imagecache)
|
||||
*
|
||||
* @param Closure $callback
|
||||
* @param int $lifetime
|
||||
* @param boolean $returnObj
|
||||
*
|
||||
* @return Image
|
||||
*/
|
||||
public function cache(Closure $callback, $lifetime = null, $returnObj = false)
|
||||
{
|
||||
if (class_exists('Intervention\\Image\\ImageCache')) {
|
||||
// create imagecache
|
||||
$imagecache = new ImageCache($this);
|
||||
|
||||
// run callback
|
||||
if (is_callable($callback)) {
|
||||
$callback($imagecache);
|
||||
}
|
||||
|
||||
return $imagecache->get($lifetime, $returnObj);
|
||||
}
|
||||
|
||||
throw new MissingDependencyException(
|
||||
"Please install package intervention/imagecache before running this function."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a driver instance according to config settings
|
||||
*
|
||||
* @return \Intervention\Image\AbstractDriver
|
||||
*/
|
||||
private function createDriver()
|
||||
{
|
||||
if (is_string($this->config['driver'])) {
|
||||
$drivername = ucfirst($this->config['driver']);
|
||||
$driverclass = sprintf('Intervention\\Image\\%s\\Driver', $drivername);
|
||||
|
||||
if (class_exists($driverclass)) {
|
||||
return new $driverclass;
|
||||
}
|
||||
|
||||
throw new NotSupportedException(
|
||||
"Driver ({$drivername}) could not be instantiated."
|
||||
);
|
||||
}
|
||||
|
||||
if ($this->config['driver'] instanceof AbstractDriver) {
|
||||
return $this->config['driver'];
|
||||
}
|
||||
|
||||
throw new NotSupportedException(
|
||||
"Unknown driver type."
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Check if all requirements are available
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
private function checkRequirements()
|
||||
{
|
||||
if ( ! function_exists('finfo_buffer')) {
|
||||
throw new MissingDependencyException(
|
||||
"PHP Fileinfo extension must be installed/enabled to use Intervention Image."
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
88
vendor/intervention/image/src/Intervention/Image/ImageManagerStatic.php
vendored
Normal file
88
vendor/intervention/image/src/Intervention/Image/ImageManagerStatic.php
vendored
Normal file
@ -0,0 +1,88 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Closure;
|
||||
|
||||
class ImageManagerStatic
|
||||
{
|
||||
/**
|
||||
* Instance of Intervention\Image\ImageManager
|
||||
*
|
||||
* @var ImageManager
|
||||
*/
|
||||
public static $manager;
|
||||
|
||||
/**
|
||||
* Creates a new instance
|
||||
*
|
||||
* @param ImageManager $manager
|
||||
*/
|
||||
public function __construct(ImageManager $manager = null)
|
||||
{
|
||||
self::$manager = $manager ? $manager : new ImageManager;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get or create new ImageManager instance
|
||||
*
|
||||
* @return ImageManager
|
||||
*/
|
||||
public static function getManager()
|
||||
{
|
||||
return self::$manager ? self::$manager : new ImageManager;
|
||||
}
|
||||
|
||||
/**
|
||||
* Statically create new custom configured image manager
|
||||
*
|
||||
* @param array $config
|
||||
*
|
||||
* @return ImageManager
|
||||
*/
|
||||
public static function configure(array $config = [])
|
||||
{
|
||||
return self::$manager = self::getManager()->configure($config);
|
||||
}
|
||||
|
||||
/**
|
||||
* Statically initiates an Image instance from different input types
|
||||
*
|
||||
* @param mixed $data
|
||||
*
|
||||
* @return \Intervention\Image\Image
|
||||
* @throws \Intervention\Image\Exception\NotReadableException
|
||||
*/
|
||||
public static function make($data)
|
||||
{
|
||||
return self::getManager()->make($data);
|
||||
}
|
||||
|
||||
/**
|
||||
* Statically creates an empty image canvas
|
||||
*
|
||||
* @param int $width
|
||||
* @param int $height
|
||||
* @param mixed $background
|
||||
*
|
||||
* @return \Intervention\Image\Image
|
||||
*/
|
||||
public static function canvas($width, $height, $background = null)
|
||||
{
|
||||
return self::getManager()->canvas($width, $height, $background);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create new cached image and run callback statically
|
||||
*
|
||||
* @param Closure $callback
|
||||
* @param int $lifetime
|
||||
* @param boolean $returnObj
|
||||
*
|
||||
* @return mixed
|
||||
*/
|
||||
public static function cache(Closure $callback, $lifetime = null, $returnObj = false)
|
||||
{
|
||||
return self::getManager()->cache($callback, $lifetime, $returnObj);
|
||||
}
|
||||
}
|
87
vendor/intervention/image/src/Intervention/Image/ImageServiceProvider.php
vendored
Normal file
87
vendor/intervention/image/src/Intervention/Image/ImageServiceProvider.php
vendored
Normal file
@ -0,0 +1,87 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Illuminate\Support\ServiceProvider;
|
||||
use Laravel\Lumen\Application as LumenApplication;
|
||||
use Illuminate\Foundation\Application as IlluminateApplication;
|
||||
|
||||
class ImageServiceProvider extends ServiceProvider
|
||||
{
|
||||
/**
|
||||
* Indicates if loading of the provider is deferred.
|
||||
*
|
||||
* @var bool
|
||||
*/
|
||||
protected $defer = false;
|
||||
|
||||
/**
|
||||
* Actual provider
|
||||
*
|
||||
* @var \Illuminate\Support\ServiceProvider
|
||||
*/
|
||||
protected $provider;
|
||||
|
||||
/**
|
||||
* Create a new service provider instance.
|
||||
*
|
||||
* @param \Illuminate\Contracts\Foundation\Application $app
|
||||
* @return void
|
||||
*/
|
||||
public function __construct($app)
|
||||
{
|
||||
parent::__construct($app);
|
||||
|
||||
$this->provider = $this->getProvider();
|
||||
}
|
||||
|
||||
/**
|
||||
* Bootstrap the application events.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function boot()
|
||||
{
|
||||
if (method_exists($this->provider, 'boot')) {
|
||||
return $this->provider->boot();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Register the service provider.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function register()
|
||||
{
|
||||
return $this->provider->register();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return ServiceProvider according to Laravel version
|
||||
*
|
||||
* @return \Intervention\Image\Provider\ProviderInterface
|
||||
*/
|
||||
private function getProvider()
|
||||
{
|
||||
if ($this->app instanceof LumenApplication) {
|
||||
$provider = '\Intervention\Image\ImageServiceProviderLumen';
|
||||
} elseif (version_compare(IlluminateApplication::VERSION, '5.0', '<')) {
|
||||
$provider = '\Intervention\Image\ImageServiceProviderLaravel4';
|
||||
} else {
|
||||
$provider = '\Intervention\Image\ImageServiceProviderLaravelRecent';
|
||||
}
|
||||
|
||||
return new $provider($this->app);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the services provided by the provider.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function provides()
|
||||
{
|
||||
return ['image'];
|
||||
}
|
||||
}
|
112
vendor/intervention/image/src/Intervention/Image/ImageServiceProviderLaravel4.php
vendored
Executable file
112
vendor/intervention/image/src/Intervention/Image/ImageServiceProviderLaravel4.php
vendored
Executable file
@ -0,0 +1,112 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Illuminate\Support\ServiceProvider;
|
||||
use Illuminate\Http\Response as IlluminateResponse;
|
||||
|
||||
class ImageServiceProviderLaravel4 extends ServiceProvider
|
||||
{
|
||||
/**
|
||||
* Bootstrap the application events.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function boot()
|
||||
{
|
||||
$this->package('intervention/image');
|
||||
|
||||
// try to create imagecache route only if imagecache is present
|
||||
if (class_exists('Intervention\\Image\\ImageCache')) {
|
||||
|
||||
$app = $this->app;
|
||||
|
||||
// load imagecache config
|
||||
$app['config']->package('intervention/imagecache', __DIR__.'/../../../../imagecache/src/config', 'imagecache');
|
||||
$config = $app['config'];
|
||||
|
||||
// create dynamic manipulation route
|
||||
if (is_string($config->get('imagecache::route'))) {
|
||||
|
||||
// add original to route templates
|
||||
$config->set('imagecache::templates.original', null);
|
||||
|
||||
// setup image manipulator route
|
||||
$app['router']->get($config->get('imagecache::route').'/{template}/{filename}', ['as' => 'imagecache', function ($template, $filename) use ($app, $config) {
|
||||
|
||||
// disable session cookies for image route
|
||||
$app['config']->set('session.driver', 'array');
|
||||
|
||||
// find file
|
||||
foreach ($config->get('imagecache::paths') as $path) {
|
||||
// don't allow '..' in filenames
|
||||
$image_path = $path.'/'.str_replace('..', '', $filename);
|
||||
if (file_exists($image_path) && is_file($image_path)) {
|
||||
break;
|
||||
} else {
|
||||
$image_path = false;
|
||||
}
|
||||
}
|
||||
|
||||
// abort if file not found
|
||||
if ($image_path === false) {
|
||||
$app->abort(404);
|
||||
}
|
||||
|
||||
// define template callback
|
||||
$callback = $config->get("imagecache::templates.{$template}");
|
||||
|
||||
if (is_callable($callback) || class_exists($callback)) {
|
||||
|
||||
// image manipulation based on callback
|
||||
$content = $app['image']->cache(function ($image) use ($image_path, $callback) {
|
||||
|
||||
switch (true) {
|
||||
case is_callable($callback):
|
||||
return $callback($image->make($image_path));
|
||||
break;
|
||||
|
||||
case class_exists($callback):
|
||||
return $image->make($image_path)->filter(new $callback);
|
||||
break;
|
||||
}
|
||||
|
||||
}, $config->get('imagecache::lifetime'));
|
||||
|
||||
} else {
|
||||
|
||||
// get original image file contents
|
||||
$content = file_get_contents($image_path);
|
||||
}
|
||||
|
||||
// define mime type
|
||||
$mime = finfo_buffer(finfo_open(FILEINFO_MIME_TYPE), $content);
|
||||
|
||||
// return http response
|
||||
return new IlluminateResponse($content, 200, [
|
||||
'Content-Type' => $mime,
|
||||
'Cache-Control' => 'max-age='.($config->get('imagecache::lifetime')*60).', public',
|
||||
'Etag' => md5($content)
|
||||
]);
|
||||
|
||||
}])->where(['template' => join('|', array_keys($config->get('imagecache::templates'))), 'filename' => '[ \w\\.\\/\\-]+']);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Register the service provider.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function register()
|
||||
{
|
||||
$app = $this->app;
|
||||
|
||||
$app['image'] = $app->share(function ($app) {
|
||||
return new ImageManager($app['config']->get('image::config'));
|
||||
});
|
||||
|
||||
$app->alias('image', 'Intervention\Image\ImageManager');
|
||||
}
|
||||
}
|
106
vendor/intervention/image/src/Intervention/Image/ImageServiceProviderLaravelRecent.php
vendored
Normal file
106
vendor/intervention/image/src/Intervention/Image/ImageServiceProviderLaravelRecent.php
vendored
Normal file
@ -0,0 +1,106 @@
|
||||
<?php
|
||||
|
||||
namespace Intervention\Image;
|
||||
|
||||
use Illuminate\Support\ServiceProvider;
|
||||
|
||||
class ImageServiceProviderLaravelRecent extends ServiceProvider
|
||||
{
|
||||
/**
|
||||
* Determines if Intervention Imagecache is installed
|
||||
*
|
||||
* @return boolean
|
||||
*/
|
||||
private function cacheIsInstalled()
|
||||
{
|
||||
return class_exists('Intervention\\Image\\ImageCache');
|
||||
}
|
||||
|
||||
/**
|
||||
* Bootstrap the application events.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function boot()
|
||||
{
|
||||
$this->publishes([
|
||||
__DIR__.'/../../config/config.php' => config_path('image.php')
|
||||
]);
|
||||
|
||||
// setup intervention/imagecache if package is installed
|
||||
$this->cacheIsInstalled() ? $this->bootstrapImageCache() : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Register the service provider.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function register()
|
||||
{
|
||||
$app = $this->app;
|
||||
|
||||
// merge default config
|
||||
$this->mergeConfigFrom(
|
||||
__DIR__.'/../../config/config.php',
|
||||
'image'
|
||||
);
|
||||
|
||||
// create image
|
||||
$app->singleton('image', function ($app) {
|
||||
return new ImageManager($this->getImageConfig($app));
|
||||
});
|
||||
|
||||
$app->alias('image', 'Intervention\Image\ImageManager');
|
||||
}
|
||||
|
||||
/**
|
||||
* Bootstrap imagecache
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
protected function bootstrapImageCache()
|
||||
{
|
||||
$app = $this->app;
|
||||
$config = __DIR__.'/../../../../imagecache/src/config/config.php';
|
||||
|
||||
$this->publishes([
|
||||
$config => config_path('imagecache.php')
|
||||
]);
|
||||
|
||||
// merge default config
|
||||
$this->mergeConfigFrom(
|
||||
$config,
|
||||
'imagecache'
|
||||
);
|
||||
|
||||
// imagecache route
|
||||
if (is_string(config('imagecache.route'))) {
|
||||
|
||||
$filename_pattern = '[ \w\\.\\/\\-\\@\(\)\=]+';
|
||||
|
||||
// route to access template applied image file
|
||||
$app['router']->get(config('imagecache.route').'/{template}/{filename}', [
|
||||
'uses' => 'Intervention\Image\ImageCacheController@getResponse',
|
||||
'as' => 'imagecache'
|
||||
])->where(['filename' => $filename_pattern]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Return image configuration as array
|
||||
*
|
||||
* @param Application $app
|
||||
* @return array
|
||||
*/
|
||||
private function getImageConfig($app)
|
||||
{
|
||||
$config = $app['config']->get('image');
|
||||
|
||||
if (is_null($config)) {
|
||||
return [];
|
||||
}
|
||||
|
||||
return $config;
|
||||
}
|
||||
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user