Product.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. <?php
  2. namespace ModulesGarden\Servers\ProxmoxVps\Packages\WhmcsService;
  3. use ModulesGarden\Servers\ProxmoxVps\Core\HandlerError\Exceptions\Exception;
  4. use ModulesGarden\Servers\ProxmoxVps\Core\HandlerError\ErrorCodes\ErrorCodesLib;
  5. use ModulesGarden\Servers\ProxmoxVps\Core\Models\ProductSettings\Repository;
  6. use ModulesGarden\Servers\ProxmoxVps\Core\Models\Whmcs\Currency;
  7. use ModulesGarden\Servers\ProxmoxVps\Packages\WhmcsService\Config\Enum;
  8. use ModulesGarden\Servers\ProxmoxVps\Core\Traits\AppParams;
  9. use ModulesGarden\Servers\ProxmoxVps\Packages\WhmcsService\Traits\ConfigurableOptionsConfig;
  10. use ModulesGarden\Servers\ProxmoxVps\Packages\WhmcsService\Traits\CustomFieldsConfig;
  11. use WHMCS\Database\Capsule;
  12. class Product
  13. {
  14. use ConfigurableOptionsConfig;
  15. use CustomFieldsConfig;
  16. use AppParams;
  17. protected $productId = null;
  18. protected $product = null;
  19. protected $configOptions = null;
  20. protected $configurableOptionsGroupCreated = false;
  21. public function __construct($productId = null)
  22. {
  23. $this->setProductId($productId);
  24. $this->loadWhmcsService();
  25. }
  26. protected function setProductId($productId = null)
  27. {
  28. $relationId = (int)$productId;
  29. if ($relationId <= 0)
  30. {
  31. throw new Exception(ErrorCodesLib::CORE_WS_000002);
  32. }
  33. $this->productId = $relationId;
  34. }
  35. protected function loadWhmcsService()
  36. {
  37. $this->product = \WHMCS\Product\Product::find($this->productId);
  38. }
  39. public function loadConfigOptions($force = false)
  40. {
  41. if ($force || $this->configOptions === null)
  42. {
  43. $this->configOptions = Capsule::table('tblproductconfigoptions')
  44. ->leftJoin('tblproductconfiglinks', 'tblproductconfigoptions.gid', '=', 'tblproductconfiglinks.gid')
  45. ->join('tblproductconfiggroups', 'tblproductconfigoptions.gid', '=', 'tblproductconfiggroups.id')
  46. ->select('tblproductconfigoptions.id', 'tblproductconfigoptions.optionname', 'tblproductconfigoptions.gid')
  47. ->where('tblproductconfiglinks.pid', $this->productId)->get();
  48. }
  49. }
  50. public function doesConfigurableOptionExist($optionName = null)
  51. {
  52. if (!is_string($optionName) || trim($optionName) === '')
  53. {
  54. return false;
  55. }
  56. $this->loadConfigOptions();
  57. $rawOptionName = $this->configOptionNameToRaw($optionName);
  58. foreach ($this->configOptions as $option)
  59. {
  60. if ($rawOptionName === $this->configOptionNameToRaw($option->optionname))
  61. {
  62. return true;
  63. }
  64. }
  65. return false;
  66. }
  67. public function getOptionGroupId($optionName = null)
  68. {
  69. if (!is_string($optionName) || trim($optionName) === '')
  70. {
  71. return false;
  72. }
  73. $this->loadConfigOptions();
  74. $rawOptionName = $this->configOptionNameToRaw($optionName);
  75. foreach ($this->configOptions as $option)
  76. {
  77. if ($rawOptionName === $this->configOptionNameToRaw($option->optionname))
  78. {
  79. return $option->gid;
  80. }
  81. }
  82. return false;
  83. }
  84. public function getFirstOptionGroupId()
  85. {
  86. $this->loadConfigOptions();
  87. foreach ($this->configOptions as $option)
  88. {
  89. return $option->gid;
  90. }
  91. $optionGroup = Capsule::table('tblproductconfiggroups')
  92. ->join('tblproductconfiglinks', 'tblproductconfiglinks.gid', '=', 'tblproductconfiggroups.id')
  93. ->where('tblproductconfiglinks.pid', $this->productId)->first();
  94. if (is_object($optionGroup))
  95. {
  96. return $optionGroup->gid;
  97. }
  98. return false;
  99. }
  100. public function createConfigOptionsGroup()
  101. {
  102. $productName = $this->product->name;
  103. $moduleName = $this->getAppParam('name');
  104. $groupId = Capsule::table('tblproductconfiggroups')->insertGetId([
  105. 'name' => 'Configurable options for ' . $productName . ' product',
  106. 'description' => 'Auto generated by module ' . $moduleName
  107. ]);
  108. if (is_int($groupId) && $groupId > 0)
  109. {
  110. Capsule::table('tblproductconfiglinks')->insertGetId(['gid' => $groupId, 'pid' => $this->productId]);
  111. }
  112. $this->configurableOptionsGroupCreated = true;
  113. }
  114. public function configOptionNameToRaw($optionName)
  115. {
  116. if (!is_string($optionName) || trim($optionName) === '' || substr($optionName, 0, 1) === '|')
  117. {
  118. return '';
  119. }
  120. if (strpos($optionName, '|') > 0)
  121. {
  122. $parts = explode('|', $optionName);
  123. return $parts[0];
  124. }
  125. return $optionName;
  126. }
  127. public function addConfigurableOption($optionName = null)
  128. {
  129. if (!$this->canOptionBeAdded($optionName))
  130. {
  131. return;
  132. }
  133. $gid = $this->getConfigurableOptionGroupId();
  134. $this->createConfigurableOption($optionName, $gid);
  135. }
  136. public function createConfigurableOption($optionName, $gid)
  137. {
  138. foreach ($this->configOptionsList as $option)
  139. {
  140. if ($option[Enum::OPTION_NAME] === $optionName || $this->configOptionNameToRaw($option[Enum::OPTION_NAME]) === $optionName)
  141. {
  142. $optId = Capsule::table('tblproductconfigoptions')->insertGetId([
  143. Enum::OPTION_GROUP_ID => $gid,
  144. Enum::OPTION_NAME => $option[Enum::OPTION_NAME],
  145. Enum::OPTION_TYPE => $option[Enum::OPTION_TYPE],
  146. Enum::OPTION_QUANTITY_MIN => $option[Enum::OPTION_QUANTITY_MIN] ? $option[Enum::OPTION_QUANTITY_MIN] : Enum::OPTION_QUANTITY_MIN_DEFAULT,
  147. Enum::OPTION_QUANTITY_MAX => $option[Enum::OPTION_QUANTITY_MAX] ? $option[Enum::OPTION_QUANTITY_MAX] : Enum::OPTION_QUANTITY_MAX_DEFAULT,
  148. Enum::ORDER => $option[Enum::ORDER] ? $option[Enum::ORDER] : Enum::ORDER_DEFAULT,
  149. Enum::HIDDEN => $option[Enum::HIDDEN] ? $option[Enum::HIDDEN] : Enum::HIDDEN_DEFAULT
  150. ]);
  151. if ($option[Enum::OPTION_TYPE] == Enum::OPTION_TYPE_QUANTITY && !$option[Enum::CONFIG_SUB_OPTIONS])
  152. {
  153. $defaultConfig = Enum::DEFAULT_QUANTITY_SUB_CONFIG;
  154. $this->createConfigurableSubOption($optId, $defaultConfig[Enum::OPTION_SUB_NAME],
  155. $defaultConfig[Enum::OPTION_SUB_ORDER], $defaultConfig[Enum::OPTION_SUB_HIDDEN]);
  156. }
  157. foreach ($option[Enum::CONFIG_SUB_OPTIONS] as $subOption)
  158. {
  159. $this->createConfigurableSubOption($optId, $subOption[Enum::OPTION_SUB_NAME],
  160. $subOption[Enum::OPTION_SUB_ORDER], $subOption[Enum::OPTION_SUB_HIDDEN]);
  161. }
  162. foreach ($this->getCallbackSubOptions($optionName) as $subOption)
  163. {
  164. $this->createConfigurableSubOption($optId, $subOption[Enum::OPTION_SUB_NAME],
  165. $subOption[Enum::OPTION_SUB_ORDER], $subOption[Enum::OPTION_SUB_HIDDEN]);
  166. }
  167. break;
  168. }
  169. }
  170. }
  171. public function createConfigurableSubOption($optId = null, $optName = null, $sortOrder = 0, $hidden = 0)
  172. {
  173. if (!is_string($optName) || trim($optName) === '')
  174. {
  175. return;
  176. }
  177. $subId = Capsule::table('tblproductconfigoptionssub')->insertGetId([
  178. Enum::OPTION_SUB_OPTION_ID => (int)$optId,
  179. Enum::OPTION_SUB_NAME => $optName,
  180. Enum::OPTION_SUB_ORDER => (int)$sortOrder ? (int)$sortOrder : Enum::OPTION_SUB_ORDER_DEFAULT,
  181. Enum::OPTION_SUB_HIDDEN => (int)$hidden ? (int)$hidden : Enum::OPTION_SUB_HIDDEN_DEFAULT,
  182. ]);
  183. //add default pricing
  184. $this->insertDefaultConfigOptionPricing($subId);
  185. return $subId;
  186. }
  187. public function insertDefaultConfigOptionPricing($relId = null, $defaultPrice = '0.00')
  188. {
  189. $currencyIds = Currency::pluck("id")->all();
  190. if(empty( $currencyIds)|| !$relId)
  191. {
  192. return null;
  193. }
  194. foreach ( $currencyIds as $currencyId){
  195. Capsule::table('tblpricing')->insertGetId([ 'type' => 'configoptions', 'currency' => $currencyId,
  196. 'relid' => $relId, 'msetupfee' => $defaultPrice, 'qsetupfee' => $defaultPrice, 'ssetupfee' => $defaultPrice,
  197. 'asetupfee' => $defaultPrice, 'bsetupfee' => $defaultPrice, 'tsetupfee' => $defaultPrice, 'monthly' => $defaultPrice,
  198. 'quarterly' => $defaultPrice, 'semiannually' => $defaultPrice, 'annually' => $defaultPrice,
  199. 'biennially' => $defaultPrice, 'triennially' => $defaultPrice
  200. ]);
  201. }
  202. }
  203. public function getConfigurableOptionGroupId()
  204. {
  205. $gid = $this->getFirstOptionGroupId();
  206. if (!$gid)
  207. {
  208. $this->createConfigOptionsGroup();
  209. $this->loadConfigOptions(true);
  210. $gid = $this->getFirstOptionGroupId();
  211. }
  212. return $gid;
  213. }
  214. public function canOptionBeAdded($optionName = null)
  215. {
  216. if (!is_string($optionName) || trim($optionName) === '')
  217. {
  218. return false;
  219. }
  220. if ($this->doesConfigurableOptionExist($optionName))
  221. {
  222. return false;
  223. }
  224. $configOptionParams = $this->getConfigurableOptionConfigParams($optionName);
  225. if (!$configOptionParams)
  226. {
  227. return false;
  228. }
  229. return true;
  230. }
  231. public function getCallbackSubOptions($optionName = null)
  232. {
  233. if(!class_exists('\ModulesGarden\Servers\ProxmoxVps\App\Config\Packages\WhmcsService'))
  234. {
  235. return null;
  236. }
  237. $service = new \ModulesGarden\Servers\ProxmoxVps\App\Config\Packages\WhmcsService();
  238. if (method_exists($service, 'getSuboptionsByCallback') && is_callable([$service, 'getSuboptionsByCallback']))
  239. {
  240. return $service->getSuboptionsByCallback($optionName);
  241. }
  242. return null;
  243. }
  244. public function isConfigurableOptionsGroupCreated()
  245. {
  246. return $this->configurableOptionsGroupCreated;
  247. }
  248. public function getProductConfig()
  249. {
  250. $settingRepo = new Repository();
  251. $productSettings = $settingRepo->getProductSettings($this->productId);
  252. return $productSettings;
  253. }
  254. }