local-fields.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592
  1. <?php
  2. // Register notices stores.
  3. acf_register_store( 'local-fields' );
  4. acf_register_store( 'local-groups' );
  5. acf_register_store( 'local-empty' );
  6. // Register filter.
  7. acf_enable_filter( 'local' );
  8. /**
  9. * acf_enable_local
  10. *
  11. * Enables the local filter.
  12. *
  13. * @date 22/1/19
  14. * @since 5.7.10
  15. *
  16. * @param void
  17. * @return void
  18. */
  19. function acf_enable_local() {
  20. acf_enable_filter( 'local' );
  21. }
  22. /**
  23. * acf_disable_local
  24. *
  25. * Disables the local filter.
  26. *
  27. * @date 22/1/19
  28. * @since 5.7.10
  29. *
  30. * @param void
  31. * @return void
  32. */
  33. function acf_disable_local() {
  34. acf_disable_filter( 'local' );
  35. }
  36. /**
  37. * acf_is_local_enabled
  38. *
  39. * Returns true if local fields are enabled.
  40. *
  41. * @date 23/1/19
  42. * @since 5.7.10
  43. *
  44. * @param void
  45. * @return bool
  46. */
  47. function acf_is_local_enabled() {
  48. return ( acf_is_filter_enabled( 'local' ) && acf_get_setting( 'local' ) );
  49. }
  50. /**
  51. * acf_get_local_store
  52. *
  53. * Returns either local store or a dummy store for the given name.
  54. *
  55. * @date 23/1/19
  56. * @since 5.7.10
  57. *
  58. * @param string $name The store name (fields|groups).
  59. * @return ACF_Data
  60. */
  61. function acf_get_local_store( $name = '' ) {
  62. // Check if enabled.
  63. if ( acf_is_local_enabled() ) {
  64. return acf_get_store( "local-$name" );
  65. // Return dummy store if not enabled.
  66. } else {
  67. return acf_get_store( 'local-empty' );
  68. }
  69. }
  70. /**
  71. * acf_reset_local
  72. *
  73. * Resets the local data.
  74. *
  75. * @date 22/1/19
  76. * @since 5.7.10
  77. *
  78. * @param void
  79. * @return void
  80. */
  81. function acf_reset_local() {
  82. acf_get_local_store( 'fields' )->reset();
  83. acf_get_local_store( 'groups' )->reset();
  84. }
  85. /**
  86. * acf_get_local_field_groups
  87. *
  88. * Returns all local field groups.
  89. *
  90. * @date 22/1/19
  91. * @since 5.7.10
  92. *
  93. * @param void
  94. * @return array
  95. */
  96. function acf_get_local_field_groups() {
  97. return acf_get_local_store( 'groups' )->get();
  98. }
  99. /**
  100. * acf_have_local_field_groups
  101. *
  102. * description
  103. *
  104. * @date 22/1/19
  105. * @since 5.7.10
  106. *
  107. * @param type $var Description. Default.
  108. * @return type Description.
  109. */
  110. function acf_have_local_field_groups() {
  111. return acf_get_local_store( 'groups' )->count() ? true : false;
  112. }
  113. /**
  114. * acf_count_local_field_groups
  115. *
  116. * description
  117. *
  118. * @date 22/1/19
  119. * @since 5.7.10
  120. *
  121. * @param type $var Description. Default.
  122. * @return type Description.
  123. */
  124. function acf_count_local_field_groups() {
  125. return acf_get_local_store( 'groups' )->count();
  126. }
  127. /**
  128. * acf_add_local_field_group
  129. *
  130. * Adds a local field group.
  131. *
  132. * @date 22/1/19
  133. * @since 5.7.10
  134. *
  135. * @param array $field_group The field group array.
  136. * @return bool
  137. */
  138. function acf_add_local_field_group( $field_group ) {
  139. // Apply default properties needed for import.
  140. $field_group = wp_parse_args(
  141. $field_group,
  142. array(
  143. 'key' => '',
  144. 'title' => '',
  145. 'fields' => array(),
  146. 'local' => 'php',
  147. )
  148. );
  149. // Generate key if only name is provided.
  150. if ( ! $field_group['key'] ) {
  151. $field_group['key'] = 'group_' . acf_slugify( $field_group['title'], '_' );
  152. }
  153. // Bail early if field group already exists.
  154. if ( acf_is_local_field_group( $field_group['key'] ) ) {
  155. return false;
  156. }
  157. // Prepare field group for import (adds menu_order and parent properties to fields).
  158. $field_group = acf_prepare_field_group_for_import( $field_group );
  159. // Extract fields from group.
  160. $fields = acf_extract_var( $field_group, 'fields' );
  161. // Add to store
  162. acf_get_local_store( 'groups' )->set( $field_group['key'], $field_group );
  163. // Add fields
  164. if ( $fields ) {
  165. acf_add_local_fields( $fields );
  166. }
  167. // Return true on success.
  168. return true;
  169. }
  170. /**
  171. * register_field_group
  172. *
  173. * See acf_add_local_field_group().
  174. *
  175. * @date 22/1/19
  176. * @since 5.7.10
  177. *
  178. * @param array $field_group The field group array.
  179. * @return void
  180. */
  181. function register_field_group( $field_group ) {
  182. acf_add_local_field_group( $field_group );
  183. }
  184. /**
  185. * acf_remove_local_field_group
  186. *
  187. * Removes a field group for the given key.
  188. *
  189. * @date 22/1/19
  190. * @since 5.7.10
  191. *
  192. * @param string $key The field group key.
  193. * @return bool
  194. */
  195. function acf_remove_local_field_group( $key = '' ) {
  196. return acf_get_local_store( 'groups' )->remove( $key );
  197. }
  198. /**
  199. * acf_is_local_field_group
  200. *
  201. * Returns true if a field group exists for the given key.
  202. *
  203. * @date 22/1/19
  204. * @since 5.7.10
  205. *
  206. * @param string $key The field group key.
  207. * @return bool
  208. */
  209. function acf_is_local_field_group( $key = '' ) {
  210. return acf_get_local_store( 'groups' )->has( $key );
  211. }
  212. /**
  213. * acf_is_local_field_group_key
  214. *
  215. * Returns true if a field group exists for the given key.
  216. *
  217. * @date 22/1/19
  218. * @since 5.7.10
  219. *
  220. * @param string $key The field group group key.
  221. * @return bool
  222. */
  223. function acf_is_local_field_group_key( $key = '' ) {
  224. return acf_get_local_store( 'groups' )->is( $key );
  225. }
  226. /**
  227. * acf_get_local_field_group
  228. *
  229. * Returns a field group for the given key.
  230. *
  231. * @date 22/1/19
  232. * @since 5.7.10
  233. *
  234. * @param string $key The field group key.
  235. * @return (array|null)
  236. */
  237. function acf_get_local_field_group( $key = '' ) {
  238. return acf_get_local_store( 'groups' )->get( $key );
  239. }
  240. /**
  241. * acf_add_local_fields
  242. *
  243. * Adds an array of local fields.
  244. *
  245. * @date 22/1/19
  246. * @since 5.7.10
  247. *
  248. * @param array $fields An array of un prepared fields.
  249. * @return array
  250. */
  251. function acf_add_local_fields( $fields = array() ) {
  252. // Prepare for import (allows parent fields to offer up children).
  253. $fields = acf_prepare_fields_for_import( $fields );
  254. // Add each field.
  255. foreach ( $fields as $field ) {
  256. acf_add_local_field( $field, true );
  257. }
  258. }
  259. /**
  260. * acf_get_local_fields
  261. *
  262. * Returns all local fields for the given parent.
  263. *
  264. * @date 22/1/19
  265. * @since 5.7.10
  266. *
  267. * @param string $parent The parent key.
  268. * @return array
  269. */
  270. function acf_get_local_fields( $parent = '' ) {
  271. // Return children
  272. if ( $parent ) {
  273. return acf_get_local_store( 'fields' )->query(
  274. array(
  275. 'parent' => $parent,
  276. )
  277. );
  278. // Return all.
  279. } else {
  280. return acf_get_local_store( 'fields' )->get();
  281. }
  282. }
  283. /**
  284. * acf_have_local_fields
  285. *
  286. * Returns true if local fields exist.
  287. *
  288. * @date 22/1/19
  289. * @since 5.7.10
  290. *
  291. * @param string $parent The parent key.
  292. * @return bool
  293. */
  294. function acf_have_local_fields( $parent = '' ) {
  295. return acf_get_local_fields( $parent ) ? true : false;
  296. }
  297. /**
  298. * acf_count_local_fields
  299. *
  300. * Returns the number of local fields for the given parent.
  301. *
  302. * @date 22/1/19
  303. * @since 5.7.10
  304. *
  305. * @param string $parent The parent key.
  306. * @return int
  307. */
  308. function acf_count_local_fields( $parent = '' ) {
  309. return count( acf_get_local_fields( $parent ) );
  310. }
  311. /**
  312. * acf_add_local_field
  313. *
  314. * Adds a local field.
  315. *
  316. * @date 22/1/19
  317. * @since 5.7.10
  318. *
  319. * @param array $field The field array.
  320. * @param bool $prepared Whether or not the field has already been prepared for import.
  321. * @return void
  322. */
  323. function acf_add_local_field( $field, $prepared = false ) {
  324. // Apply default properties needed for import.
  325. $field = wp_parse_args(
  326. $field,
  327. array(
  328. 'key' => '',
  329. 'name' => '',
  330. 'type' => '',
  331. 'parent' => '',
  332. )
  333. );
  334. // Generate key if only name is provided.
  335. if ( ! $field['key'] ) {
  336. $field['key'] = 'field_' . $field['name'];
  337. }
  338. // If called directly, allow sub fields to be correctly prepared.
  339. if ( ! $prepared ) {
  340. return acf_add_local_fields( array( $field ) );
  341. }
  342. // Extract attributes.
  343. $key = $field['key'];
  344. $name = $field['name'];
  345. // Allow sub field to be added multipel times to different parents.
  346. $store = acf_get_local_store( 'fields' );
  347. if ( $store->is( $key ) ) {
  348. $old_key = _acf_generate_local_key( $store->get( $key ) );
  349. $new_key = _acf_generate_local_key( $field );
  350. if ( $old_key !== $new_key ) {
  351. $key = $new_key;
  352. }
  353. }
  354. // Add field.
  355. $store->set( $key, $field )->alias( $key, $name );
  356. }
  357. /**
  358. * _acf_generate_local_key
  359. *
  360. * Generates a unique key based on the field's parent.
  361. *
  362. * @date 22/1/19
  363. * @since 5.7.10
  364. *
  365. * @param string $key The field key.
  366. * @return bool
  367. */
  368. function _acf_generate_local_key( $field ) {
  369. return "{$field['key']}:{$field['parent']}";
  370. }
  371. /**
  372. * acf_remove_local_field
  373. *
  374. * Removes a field for the given key.
  375. *
  376. * @date 22/1/19
  377. * @since 5.7.10
  378. *
  379. * @param string $key The field key.
  380. * @return bool
  381. */
  382. function acf_remove_local_field( $key = '' ) {
  383. return acf_get_local_store( 'fields' )->remove( $key );
  384. }
  385. /**
  386. * acf_is_local_field
  387. *
  388. * Returns true if a field exists for the given key or name.
  389. *
  390. * @date 22/1/19
  391. * @since 5.7.10
  392. *
  393. * @param string $key The field group key.
  394. * @return bool
  395. */
  396. function acf_is_local_field( $key = '' ) {
  397. return acf_get_local_store( 'fields' )->has( $key );
  398. }
  399. /**
  400. * acf_is_local_field_key
  401. *
  402. * Returns true if a field exists for the given key.
  403. *
  404. * @date 22/1/19
  405. * @since 5.7.10
  406. *
  407. * @param string $key The field group key.
  408. * @return bool
  409. */
  410. function acf_is_local_field_key( $key = '' ) {
  411. return acf_get_local_store( 'fields' )->is( $key );
  412. }
  413. /**
  414. * acf_get_local_field
  415. *
  416. * Returns a field for the given key.
  417. *
  418. * @date 22/1/19
  419. * @since 5.7.10
  420. *
  421. * @param string $key The field group key.
  422. * @return (array|null)
  423. */
  424. function acf_get_local_field( $key = '' ) {
  425. return acf_get_local_store( 'fields' )->get( $key );
  426. }
  427. /**
  428. * _acf_apply_get_local_field_groups
  429. *
  430. * Appends local field groups to the provided array.
  431. *
  432. * @date 23/1/19
  433. * @since 5.7.10
  434. *
  435. * @param array $field_groups An array of field groups.
  436. * @return array
  437. */
  438. function _acf_apply_get_local_field_groups( $groups = array() ) {
  439. // Get local groups
  440. $local = acf_get_local_field_groups();
  441. if ( $local ) {
  442. // Generate map of "index" => "key" data.
  443. $map = wp_list_pluck( $groups, 'key' );
  444. // Loop over groups and update/append local.
  445. foreach ( $local as $group ) {
  446. // Get group allowing cache and filters to run.
  447. // $group = acf_get_field_group( $group['key'] );
  448. // Update.
  449. $i = array_search( $group['key'], $map );
  450. if ( $i !== false ) {
  451. unset( $group['ID'] );
  452. $groups[ $i ] = array_merge( $groups[ $i ], $group );
  453. // Append
  454. } else {
  455. $groups[] = acf_get_field_group( $group['key'] );
  456. }
  457. }
  458. // Sort list via menu_order and title.
  459. $groups = wp_list_sort(
  460. $groups,
  461. array(
  462. 'menu_order' => 'ASC',
  463. 'title' => 'ASC',
  464. )
  465. );
  466. }
  467. // Return groups.
  468. return $groups;
  469. }
  470. // Hook into filter.
  471. add_filter( 'acf/load_field_groups', '_acf_apply_get_local_field_groups', 20, 1 );
  472. /**
  473. * _acf_apply_is_local_field_key
  474. *
  475. * Returns true if is a local key.
  476. *
  477. * @date 23/1/19
  478. * @since 5.7.10
  479. *
  480. * @param bool $bool The result.
  481. * @param string $id The identifier.
  482. * @return bool
  483. */
  484. function _acf_apply_is_local_field_key( $bool, $id ) {
  485. return acf_is_local_field_key( $id );
  486. }
  487. // Hook into filter.
  488. add_filter( 'acf/is_field_key', '_acf_apply_is_local_field_key', 20, 2 );
  489. /**
  490. * _acf_apply_is_local_field_group_key
  491. *
  492. * Returns true if is a local key.
  493. *
  494. * @date 23/1/19
  495. * @since 5.7.10
  496. *
  497. * @param bool $bool The result.
  498. * @param string $id The identifier.
  499. * @return bool
  500. */
  501. function _acf_apply_is_local_field_group_key( $bool, $id ) {
  502. return acf_is_local_field_group_key( $id );
  503. }
  504. // Hook into filter.
  505. add_filter( 'acf/is_field_group_key', '_acf_apply_is_local_field_group_key', 20, 2 );
  506. /**
  507. * _acf_do_prepare_local_fields
  508. *
  509. * Local fields that are added too early will not be correctly prepared by the field type class.
  510. *
  511. * @date 23/1/19
  512. * @since 5.7.10
  513. *
  514. * @param void
  515. * @return void
  516. */
  517. function _acf_do_prepare_local_fields() {
  518. // Get fields.
  519. $fields = acf_get_local_fields();
  520. // If fields have been registered early, re-add to correctly prepare them.
  521. if ( $fields ) {
  522. acf_add_local_fields( $fields );
  523. }
  524. }
  525. // Hook into action.
  526. add_action( 'acf/include_fields', '_acf_do_prepare_local_fields', 0, 1 );