Heartbeat.php 29 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009
  1. <?php
  2. namespace Drupal\heartbeat\Entity;
  3. use Drupal\Core\Entity\EntityStorageInterface;
  4. use Drupal\Core\Field\BaseFieldDefinition;
  5. use Drupal\Core\Entity\RevisionableContentEntityBase;
  6. use Drupal\Core\Entity\EntityChangedTrait;
  7. use Drupal\Core\Entity\EntityTypeInterface;
  8. use Drupal\Core\Utility\Token;
  9. use Drupal\Core\Url;
  10. use Drupal\Core\Link;
  11. use Drupal\Core\Database\Database;
  12. use Drupal\flag\FlagService;
  13. use Drupal\taxonomy\Entity\Term;
  14. use Drupal\user\Entity\User;
  15. use Drupal\user\UserInterface;
  16. /**
  17. * Defines the Heartbeat entity.
  18. *
  19. * @ingroup heartbeat
  20. *
  21. * @ContentEntityType(
  22. * id = "heartbeat",
  23. * label = @Translation("Heartbeat"),
  24. * bundle_label = @Translation("Heartbeat type"),
  25. * handlers = {
  26. * "storage" = "Drupal\heartbeat\HeartbeatStorage",
  27. * "view_builder" = "Drupal\Core\Entity\EntityViewBuilder",
  28. * "list_builder" = "Drupal\heartbeat\HeartbeatListBuilder",
  29. * "views_data" = "Drupal\heartbeat\Entity\HeartbeatViewsData",
  30. * "translation" = "Drupal\heartbeat\HeartbeatTranslationHandler",
  31. *
  32. * "form" = {
  33. * "default" = "Drupal\heartbeat\Form\HeartbeatForm",
  34. * "add" = "Drupal\heartbeat\Form\HeartbeatForm",
  35. * "edit" = "Drupal\heartbeat\Form\HeartbeatForm",
  36. * "delete" = "Drupal\heartbeat\Form\HeartbeatDeleteForm",
  37. * },
  38. * "access" = "Drupal\heartbeat\HeartbeatAccessControlHandler",
  39. * "route_provider" = {
  40. * "html" = "Drupal\heartbeat\HeartbeatHtmlRouteProvider",
  41. * },
  42. * },
  43. * base_table = "heartbeat",
  44. * data_table = "heartbeat_field_data",
  45. * revision_table = "heartbeat_revision",
  46. * revision_data_table = "heartbeat_field_revision",
  47. * translatable = TRUE,
  48. * admin_permission = "administer heartbeat entities",
  49. * entity_keys = {
  50. * "id" = "id",
  51. * "revision" = "vid",
  52. * "bundle" = "type",
  53. * "label" = "name",
  54. * "uuid" = "uuid",
  55. * "uid" = "uid",
  56. * "nid" = "nid",
  57. * "langcode" = "langcode",
  58. * "status" = "status",
  59. * },
  60. * links = {
  61. * "canonical" = "/admin/structure/heartbeat/{heartbeat}",
  62. * "add-page" = "/admin/structure/heartbeat/add",
  63. * "add-form" = "/admin/structure/heartbeat/add/{heartbeat_type}",
  64. * "edit-form" = "/admin/structure/heartbeat/{heartbeat}/edit",
  65. * "delete-form" = "/admin/structure/heartbeat/{heartbeat}/delete",
  66. * "version-history" = "/admin/structure/heartbeat/{heartbeat}/revisions",
  67. * "revision" = "/admin/structure/heartbeat/{heartbeat}/revisions/{heartbeat_revision}/view",
  68. * "revision_revert" = "/admin/structure/heartbeat/{heartbeat}/revisions/{heartbeat_revision}/revert",
  69. * "translation_revert" = "/admin/structure/heartbeat/{heartbeat}/revisions/{heartbeat_revision}/revert/{langcode}",
  70. * "revision_delete" = "/admin/structure/heartbeat/{heartbeat}/revisions/{heartbeat_revision}/delete",
  71. * "collection" = "/admin/structure/heartbeat",
  72. * },
  73. * bundle_entity_type = "heartbeat_type",
  74. * field_ui_base_route = "entity.heartbeat_type.edit_form"
  75. * )
  76. */
  77. // Always block from display
  78. const HEARTBEAT_NONE = -1;
  79. // Display only activity messages that are mine or addressed to me
  80. const HEARTBEAT_PRIVATE = 0;
  81. // Only the person that is chosen by the actor, can see the message
  82. const HEARTBEAT_PUBLIC_TO_ADDRESSEE = 1;
  83. // Display activity message of all my user relations, described in contributed modules
  84. const HEARTBEAT_PUBLIC_TO_CONNECTED = 2;
  85. // Everyone can see this activity message, unless this type of message is set to private
  86. const HEARTBEAT_PUBLIC_TO_ALL = 4;
  87. //Group Types
  88. const HEARTBEAT_GROUP_NONE = 11;
  89. const HEARTBEAT_GROUP_SINGLE = 12;
  90. const HEARTBEAT_GROUP_SUMMARY = 13;
  91. const FILE_FIELD = 'Drupal\file\Plugin\Field\FieldType\FileFieldItemList';
  92. class Heartbeat extends RevisionableContentEntityBase implements HeartbeatInterface {
  93. use EntityChangedTrait;
  94. /**
  95. * {@inheritdoc}
  96. */
  97. public static function preCreate(EntityStorageInterface $storage_controller, array &$values) {
  98. parent::preCreate($storage_controller, $values);
  99. }
  100. /**
  101. * {@inheritdoc}
  102. */
  103. public function preSave(EntityStorageInterface $storage) {
  104. parent::preSave($storage);
  105. foreach (array_keys($this->getTranslationLanguages()) as $langcode) {
  106. $translation = $this->getTranslation($langcode);
  107. // If no owner has been set explicitly, make the anonymous user the owner.
  108. if (!$translation->getOwner()) {
  109. $translation->setOwnerId(0);
  110. }
  111. }
  112. // If no revision author has been set explicitly, make the heartbeat owner the
  113. // revision author.
  114. if (!$this->getRevisionUser()) {
  115. $this->setRevisionUserId($this->getOwnerId());
  116. }
  117. }
  118. /**
  119. * {@inheritdoc}
  120. */
  121. public function getType() {
  122. return $this->bundle();
  123. }
  124. /**
  125. * {@inheritdoc}
  126. */
  127. public function getName() {
  128. return $this->get('name')->value;
  129. }
  130. /**
  131. * {@inheritdoc}
  132. */
  133. public function setName($name) {
  134. $this->set('name', $name);
  135. return $this;
  136. }
  137. /**
  138. * Gets the Heartbeat message.
  139. *
  140. * @return string
  141. * Message of the Heartbeat.
  142. */
  143. public function getMessage() {
  144. return $this->get('message');
  145. }
  146. /**
  147. * Sets the Heartbeat Message.
  148. *
  149. * @param $name
  150. * @return
  151. * @internal param string $message The Heartbeat Message
  152. */
  153. public function setMessage($message) {
  154. $this->set('message', $message);
  155. }
  156. /**
  157. * {@inheritdoc}
  158. */
  159. public function getCreatedTime() {
  160. return $this->get('created')->value;
  161. }
  162. /**
  163. * {@inheritdoc}
  164. */
  165. public function setCreatedTime($timestamp) {
  166. $this->set('created', $timestamp);
  167. return $this;
  168. }
  169. /**
  170. * {@inheritdoc}
  171. */
  172. public function getOwner() {
  173. return $this->get('uid')->entity;
  174. }
  175. /**
  176. * {@inheritdoc}
  177. */
  178. public function getOwnerId() {
  179. return $this->get('uid')->target_id;
  180. }
  181. /**
  182. * {@inheritdoc}
  183. */
  184. public function setOwnerId($uid) {
  185. $this->set('uid', $uid);
  186. return $this;
  187. }
  188. /**
  189. * {@inheritdoc}
  190. */
  191. public function setOwner(UserInterface $account) {
  192. $this->set('uid', $account->id());
  193. return $this;
  194. }
  195. /**
  196. * {@inheritdoc}
  197. */
  198. public function isPublished() {
  199. return (bool) $this->getEntityKey('status');
  200. }
  201. /**
  202. * {@inheritdoc}
  203. */
  204. public function setPublished($published) {
  205. $this->set('status', $published ? TRUE : FALSE);
  206. return $this;
  207. }
  208. /**
  209. * {@inheritdoc}
  210. */
  211. public function getRevisionCreationTime() {
  212. return $this->get('revision_timestamp')->value;
  213. }
  214. /**
  215. * {@inheritdoc}
  216. */
  217. public function setRevisionCreationTime($timestamp) {
  218. $this->set('revision_timestamp', $timestamp);
  219. return $this;
  220. }
  221. /**
  222. * {@inheritdoc}
  223. */
  224. public function getRevisionUser() {
  225. return $this->get('revision_uid')->entity;
  226. }
  227. /**
  228. * {@inheritdoc}
  229. */
  230. public function setRevisionUserId($uid) {
  231. $this->set('revision_uid', $uid);
  232. return $this;
  233. }
  234. /**
  235. * {@inheritdoc}
  236. */
  237. public static function baseFieldDefinitions(EntityTypeInterface $entity_type) {
  238. $fields = parent::baseFieldDefinitions($entity_type);
  239. $fields['uid'] = BaseFieldDefinition::create('entity_reference')
  240. ->setLabel(t('Authored by'))
  241. ->setDescription(t('The user ID of author of the Heartbeat entity.'))
  242. ->setRevisionable(TRUE)
  243. ->setSetting('target_type', 'user')
  244. ->setSetting('handler', 'default')
  245. ->setTranslatable(TRUE)
  246. ->setDisplayOptions('view', array(
  247. 'label' => 'hidden',
  248. 'type' => 'author',
  249. 'weight' => 0,
  250. ))
  251. ->setDisplayOptions('form', array(
  252. 'type' => 'entity_reference_autocomplete',
  253. 'weight' => 5,
  254. 'settings' => array(
  255. 'match_operator' => 'CONTAINS',
  256. 'size' => '60',
  257. 'autocomplete_type' => 'tags',
  258. 'placeholder' => '',
  259. ),
  260. ))
  261. ->setDisplayConfigurable('form', TRUE)
  262. ->setDisplayConfigurable('view', TRUE);
  263. $fields['nid'] = BaseFieldDefinition::create('entity_reference')
  264. ->setLabel(t('Node'))
  265. ->setDescription(t('The content associated with this Heartbeat'))
  266. ->setSetting('target_type', 'node')
  267. ->setSetting('handler', 'default')
  268. ->setDisplayOptions('view', array(
  269. 'label' => 'hidden',
  270. 'type' => 'content',
  271. 'weight' => 0,
  272. ))
  273. ->setDisplayConfigurable('view', TRUE)
  274. ->setRevisionable(TRUE);
  275. $fields['name'] = BaseFieldDefinition::create('string')
  276. ->setLabel(t('Name'))
  277. ->setDescription(t('The name of the Heartbeat entity.'))
  278. ->setRevisionable(TRUE)
  279. ->setSettings(array(
  280. 'max_length' => 128,
  281. 'text_processing' => 0,
  282. ))
  283. ->setDefaultValue('')
  284. ->setDisplayOptions('view', array(
  285. 'label' => 'above',
  286. 'type' => 'string',
  287. 'weight' => -4,
  288. ))
  289. // ->setDisplayOptions('form', array(
  290. // 'type' => 'string_textfield',
  291. // 'weight' => -4,
  292. // ))
  293. ->setDisplayConfigurable('form', TRUE)
  294. ->setDisplayConfigurable('view', TRUE);
  295. $fields['message'] = BaseFieldDefinition::create('string_long')
  296. ->setLabel(t('Message'))
  297. ->setDescription(t('The message of the Heartbeat entity.'))
  298. ->setRevisionable(TRUE)
  299. ->setDisplayOptions('view', array(
  300. 'label' => 'above',
  301. 'type' => 'full_html',
  302. 'weight' => -4,
  303. ))
  304. ->setDisplayConfigurable('form', TRUE)
  305. ->setDisplayConfigurable('view', TRUE);
  306. $fields['comments'] = BaseFieldDefinition::create('comment')
  307. ->setLabel(t('Kommentare'))
  308. ->setDescription(t('Kommentare.'))
  309. ->setCardinality(BaseFieldDefinition::CARDINALITY_UNLIMITED)
  310. ->setSettings(
  311. array(
  312. 'default_mode'=> 1,
  313. 'per_page'=>50,
  314. 'anonymous'=> 0,
  315. 'form_location'=>1,
  316. 'preview'=> 1,
  317. 'comment_type'=>'heartbeat_comment',
  318. 'locked'=>false,
  319. ))
  320. ->setDefaultValue(
  321. array(
  322. 'status'=>2,
  323. 'cid'=>0,
  324. 'last_comment_timestamp'=> 0,
  325. 'last_comment_name'=> null,
  326. 'last_comment_uid'=> 0,
  327. 'comment_count'=> 0,
  328. )
  329. )
  330. ->setDisplayOptions('form', array(
  331. 'type' => 'comment_default',
  332. 'settings' => array(
  333. 'form_location' => 1,
  334. 'default_mode'=> 1,
  335. 'per_page'=>50,
  336. 'anonymous'=> 0,
  337. 'preview'=> 1,
  338. 'comment_type'=>'heartbeat_comment',
  339. 'locked'=>false,
  340. ),
  341. 'weight' => 1,
  342. ))
  343. ->setDisplayConfigurable('form', TRUE)
  344. ->setDisplayConfigurable('view', TRUE);
  345. $fields['status'] = BaseFieldDefinition::create('boolean')
  346. ->setLabel(t('Publishing status'))
  347. ->setDescription(t('A boolean indicating whether the Heartbeat is published.'))
  348. ->setRevisionable(TRUE)
  349. ->setDefaultValue(TRUE);
  350. $fields['created'] = BaseFieldDefinition::create('created')
  351. ->setLabel(t('Created'))
  352. ->setDescription(t('The time that the entity was created.'));
  353. $fields['changed'] = BaseFieldDefinition::create('changed')
  354. ->setLabel(t('Changed'))
  355. ->setDescription(t('The time that the entity was last edited.'));
  356. $fields['revision_timestamp'] = BaseFieldDefinition::create('created')
  357. ->setLabel(t('Revision timestamp'))
  358. ->setDescription(t('The time that the current revision was created.'))
  359. ->setQueryable(FALSE)
  360. ->setRevisionable(TRUE);
  361. $fields['revision_uid'] = BaseFieldDefinition::create('entity_reference')
  362. ->setLabel(t('Revision user ID'))
  363. ->setDescription(t('The user ID of the author of the current revision.'))
  364. ->setSetting('target_type', 'user')
  365. ->setQueryable(FALSE)
  366. ->setRevisionable(TRUE);
  367. $fields['revision_translation_affected'] = BaseFieldDefinition::create('boolean')
  368. ->setLabel(t('Revision translation affected'))
  369. ->setDescription(t('Indicates if the last edit of a translation belongs to current revision.'))
  370. ->setReadOnly(TRUE)
  371. ->setRevisionable(TRUE)
  372. ->setTranslatable(TRUE);
  373. return $fields;
  374. }
  375. /**
  376. * Returns the node type label for the passed node.
  377. *
  378. * @param \Drupal\heartbeat\Entity\HeartbeatInterface $heartbeat
  379. * A heartbeat entity to return the heartbeat type's label for.
  380. *
  381. * @return string|false
  382. * The heartbeat type label or FALSE if the heartbeat type is not found.
  383. *
  384. */
  385. public function heartbeat_get_type(HeartbeatInterface $heartbeat) {
  386. $type = HeartbeatType::load($heartbeat->bundle());
  387. return $type ? $type->label() : FALSE;
  388. }
  389. /**
  390. * Updates all heartbeat activities of one type to be of another type.
  391. *
  392. * @param string $old_id
  393. * The current heartbeat type of the activities.
  394. * @param string $new_id
  395. * The new heartbeat type of the activities.
  396. *
  397. * @return
  398. * The number of activities whose heartbeat type field was modified.
  399. */
  400. public function heartbeat_type_update_nodes($old_id, $new_id) {
  401. return \Drupal::entityManager()->getStorage('heartbeat')->updateType($old_id, $new_id);
  402. }
  403. /**
  404. * Builds a message template for a given HeartbeatType
  405. *
  406. * @param HeartbeatType $heartbeatType
  407. * @param null $mediaData
  408. * @return null|string
  409. */
  410. public static function buildMessage(Token $tokenService, $preparsedMessage, $entities = NULL, $entityType, $mediaData = NULL) {
  411. $arbitrarious = 'nothing at all';
  412. $naul = 'nullll';
  413. $preparsedMessage = self::wrapOwner($preparsedMessage, $entities);
  414. switch (true) {
  415. case $entityType === 'node':
  416. $parsedMessage = $tokenService->replace($preparsedMessage, $entities);
  417. if (strpos($parsedMessage, '#')) {
  418. self::parseHashtags($parsedMessage);
  419. }
  420. if (strpos($parsedMessage, '@')) {
  421. self::parseUsernames($parsedMessage);
  422. }
  423. /** @noinspection NestedTernaryOperatorInspection */
  424. $message = $parsedMessage;
  425. $message .= $mediaData ? self::buildMediaMarkup($mediaData) : '';
  426. return $message;
  427. break;
  428. case $entityType === 'status':
  429. $parsedMessage = $tokenService->replace($preparsedMessage . '<a class="status-post" href="/admin/structure/' . $entityType . '/[' . $entityType . ':id]">', $entities);
  430. if (strpos($parsedMessage, '#')) {
  431. self::parseHashtags($parsedMessage);
  432. }
  433. if (strpos($parsedMessage, '@')) {
  434. self::parseUsernames($parsedMessage);
  435. }
  436. /** @noinspection NestedTernaryOperatorInspection */
  437. $message = $parsedMessage;
  438. $message .= $mediaData ? self::buildMediaMarkup($mediaData) : 'Post';
  439. $message .= '</a>';
  440. return $message;
  441. break;
  442. case $entityType === 'user':
  443. break;
  444. case $entityType === 'flag':
  445. $returnMessage = self::handleMultipleEntities($tokenService, $preparsedMessage, $entities);
  446. return strlen($returnMessage) > 0 ? $returnMessage : "Error creating message";
  447. break;
  448. }
  449. }
  450. private static function buildMediaMarkup($mediaData) {
  451. $markup = '';
  452. foreach ($mediaData as $media) {
  453. $markup .= self::mediaTag($media->type, $media->path);
  454. }
  455. return $markup;
  456. }
  457. private static function mediaTag($type, $filePath) {
  458. if ($type == 'image') {
  459. $type = 'img';
  460. return '<' . $type . ' src="' . str_replace('public://', '/sites/default/files/', $filePath) . '" class="heartbeat-image" / >';
  461. } else if ($type == 'youtube') {
  462. $filePath = str_replace('youtube://', 'https://www.youtube.com/embed/', $filePath);
  463. return '<iframe class="heartbeat-youtube" width="auto" height="auto" src="' . $filePath . '" frameborder="0"></iframe>';
  464. } else if ($type == 'video') {
  465. return '<' . $type . ' controls src="' . str_replace('public://', '/sites/default/files/', $filePath) . '" class="heartbeat-video"></' . $type . '>';
  466. }
  467. }
  468. protected static function handleMultipleEntities(Token $tokenService, $message, $entities) {
  469. $tokens = $tokenService->scan($message);
  470. foreach($tokens as $key => $token) {
  471. foreach ($token as $type) {
  472. if (substr_count($message, $type) > 1) {
  473. foreach ($entities as $entityKey => $entityValue) {
  474. if ($entityValue instanceof \stdClass && count($entityValue->entities) > 1) {
  475. if ($key == $entityValue->type) {
  476. $messageArray = explode($type, $message);
  477. $stringRebuild = array();
  478. $replacements = array();
  479. $i = 0;
  480. foreach ($entityValue->entities as $entity) {
  481. $stringRebuild[] = $tokenService->replace($message, array($key => $entity));
  482. foreach (self::getWordRepeats($stringRebuild[$i]) as $word => $num) {
  483. if ($num > 1 && !strpos($messageArray[1], $word)) {
  484. $replacements[] = $word;
  485. }
  486. }
  487. $i++;
  488. }
  489. if (count($replacements) == 2) {
  490. $uid = $entityValue->entities[0]->id();
  491. $uid_target = $entityValue->entities[1]->id();
  492. $query = Database::getConnection()->query('
  493. SELECT status
  494. FROM heartbeat_friendship
  495. WHERE uid = :uid AND uid_target = :uid_target', array(
  496. ':uid' => $uid,
  497. ':uid_target' => $uid_target
  498. )
  499. );
  500. if ($query->fetchCol()[0] < 1) {
  501. $messageArray[1] = ' has requested friendship with ';
  502. }
  503. $user1Link = Link::fromTextAndUrl($replacements[0], $entityValue->entities[0]->toUrl());
  504. $user2Link = Link::fromTextAndUrl($replacements[1], $entityValue->entities[1]->toUrl());
  505. $rebuiltMessage = $user1Link->toString() . $messageArray[1] . $user2Link->toString();
  506. return $rebuiltMessage;
  507. }
  508. }
  509. }
  510. }
  511. }
  512. }
  513. }
  514. return null;
  515. }
  516. public static function parseHashtags(&$message) {
  517. $lastRow = false;
  518. $tagsArray = explode('#', $message);
  519. $i = 0;
  520. $num = count($tagsArray);
  521. foreach ($tagsArray as $hashtag) {
  522. if ($i === $num - 1) {
  523. $lastTagArray = explode(' ', $hashtag);
  524. if (strlen($lastTagArray[1]) > 1) {
  525. $hashtag = trim($lastTagArray[0]);
  526. $lastRow = true;
  527. $remainder = '';
  528. $lastRowArgCount = count($lastTagArray);
  529. for ($x = 1; $x < $lastRowArgCount; $x++) {
  530. $remainder .= ' ' . $lastTagArray[$x];
  531. }
  532. }
  533. }
  534. $realHashtag = strpos($hashtag, ' ') ? substr($hashtag, 0, strpos($hashtag, ' ')) : $hashtag;
  535. $tid = \Drupal::entityQuery("taxonomy_term")
  536. ->condition("name", trim($realHashtag))
  537. ->condition('vid', [
  538. 'twitter', 'tags', 'kekistan'
  539. ], 'IN')
  540. ->execute();
  541. if (count($tid) > 0) {
  542. $term = Term::load(array_values($tid)[0]);
  543. $link = Link::fromTextAndUrl('#' . $realHashtag, $term->toUrl());
  544. $lenComparison = strlen($hashtag) === strlen($realHashtag);
  545. $textToAddBack = $link->toString() . str_replace($realHashtag, '', $hashtag);
  546. $tagsArray[$i] = '<div class="heartbeat-hashtag">';
  547. $tagsArray[$i] .= !$lastRow ? $textToAddBack . ' </div>' : $textToAddBack . '</div>' . $remainder;
  548. }
  549. $i++;
  550. }
  551. $message = '';
  552. foreach ($tagsArray as $replacements) {
  553. $message .= $replacements;
  554. }
  555. }
  556. public static function parseUsernames(&$message) {
  557. $lastRow = false;
  558. $usersArray = explode('@', $message);
  559. $i = 0;
  560. $num = count($usersArray);
  561. foreach ($usersArray as $username) {
  562. if ($i === $num - 1) {
  563. $lastuserArray = explode(' ', $username);
  564. if (strlen($lastuserArray[1]) > 1) {
  565. $username = trim($lastuserArray[0]);
  566. $lastRow = true;
  567. $remainder = '';
  568. $lastRowArgCount = count($lastuserArray);
  569. for ($x = 1; $x < $lastRowArgCount; $x++) {
  570. $remainder .= ' ' . $lastuserArray[$x];
  571. }
  572. }
  573. }
  574. $tid = \Drupal::entityQuery("taxonomy_term")
  575. ->condition("name", trim($username))
  576. ->condition('vid', [
  577. 'twitter_user', 'usernames'
  578. ], 'IN'
  579. )
  580. ->execute();
  581. if (count($tid) > 0) {
  582. $term = Term::load(array_values($tid)[0]);
  583. $link = Link::fromTextAndUrl('@' . $username, $term->toUrl());
  584. $usersArray[$i] = '<div class="heartbeat-username">';
  585. $usersArray[$i] .= !$lastRow ? $link->toString() . ' </div>' : $link->toString() . '</div>' . $remainder;
  586. }
  587. $i++;
  588. }
  589. $message = '';
  590. foreach ($usersArray as $replacements) {
  591. $message .= $replacements;
  592. }
  593. }
  594. /**
  595. * Helper method to identify the number of times a word is repeated in a phrase
  596. *
  597. * @param $phrase
  598. * @return array
  599. */
  600. public static function getWordRepeats($phrase) {
  601. $counts = array();
  602. $words = explode(' ', $phrase);
  603. foreach ($words as $word) {
  604. if (!array_key_exists($word, $counts)) {
  605. $counts[$word] = 0;
  606. }
  607. $word = preg_replace("#[^a-zA-Z\-]#", "", $word);
  608. ++$counts[$word];
  609. }
  610. return $counts;
  611. }
  612. /**
  613. * Returns class of argument
  614. *
  615. * @param $field
  616. * @return string
  617. */
  618. public static function findClass($field) {
  619. return get_class($field);
  620. }
  621. /**
  622. * Returns an array of classes for array argument
  623. * @param $fields
  624. * @return array
  625. */
  626. public static function findAllMedia($fields) {
  627. return array_map(array(get_called_class(), 'findClass'), $fields);
  628. }
  629. /**
  630. * Returns all media types for an array of fields
  631. *
  632. * @param $fields
  633. * @return array
  634. * @throws \Drupal\Core\Entity\Exception\UndefinedLinkTemplateException
  635. * @throws \Drupal\Core\Entity\EntityMalformedException
  636. */
  637. public static function mediaFieldTypes($fields) {
  638. $types = array();
  639. foreach ($fields as $field) {
  640. if ($field instanceof \Drupal\file\Plugin\Field\FieldType\FileFieldItemList) {
  641. if ($field->getFieldDefinition()->getType() === 'image' ||
  642. $field->getFieldDefinition()->getType() === 'video' ||
  643. $field->getFieldDefinition()->getType() === 'audio') {
  644. $fieldValue = $field->getValue();
  645. foreach ($fieldValue as $value) {
  646. $file = \Drupal::entityTypeManager()->getStorage('file')->load($value['target_id']);
  647. if ($file !== NULL && is_object($file)) {
  648. $url = Url::fromUri($file->getFileUri());
  649. $posfind = strpos($url->getUri(), 'youtube://');
  650. if ($posfind !== 0 && $posfind === false) {
  651. $mediaObject = self::createHeartbeatMedia($field->getFieldDefinition()->getType(), $url->getUri());
  652. } else {
  653. $mediaObject = self::createHeartbeatMedia('youtube', $url->getUri());
  654. }
  655. $types[] = $mediaObject;
  656. } else {
  657. continue;
  658. }
  659. }
  660. }
  661. }
  662. }
  663. return $types;
  664. }
  665. /**
  666. * Parses a HeartbeatType message template and maps
  667. * variable values onto matching keywords
  668. *
  669. * @param $translatedMessage
  670. * @param $variables
  671. * @return string
  672. */
  673. public static function parseMessage($translatedMessage, $variables) {
  674. return strtr($translatedMessage, $variables);
  675. }
  676. public static function createHeartbeatMedia($type, $path) {
  677. $mediaObject = new \stdClass();
  678. $mediaObject->type = $type;
  679. $mediaObject->path = $path;
  680. return $mediaObject;
  681. }
  682. public static function getEntityNames($entityTypes) {
  683. $names = array();
  684. foreach ($entityTypes as $type) {
  685. if (($type->getBaseTable() === 'node') ||
  686. ($type->getBaseTable() === 'user') ||
  687. ($type->getBaseTable() === 'status')
  688. ||
  689. ($type->getStorageClass() !== NULL &&
  690. strpos($type->getStorageClass(), $type->getLabel()->getUntranslatedString())
  691. )
  692. ) {
  693. $names[] = $type->id();
  694. }
  695. }
  696. sort($names);
  697. return $names;
  698. }
  699. private static function wrapOwner($message, $entities) {
  700. foreach ($entities as $entity) {
  701. if ($entity instanceof User) {
  702. return str_replace(
  703. '[user:account-name]',
  704. '<a class="heartbeat-user" href="user/' . $entity->id() . '">[user:account-name]</a>', $message);
  705. }
  706. }
  707. return $message;
  708. }
  709. public static function flagAjaxMarkup($flagId, $entity, FlagService $flagService) {
  710. $flag = $flagService->getFlagById($flagId);
  711. $link = $flag->getLinkTypePlugin()->getAsLink($flag, $entity);
  712. $options = $link->getUrl()->getOptions();
  713. $options['query']['destination'] = 'node';
  714. $link->getUrl()->setOptions($options);
  715. $action = $flag->getLinkTypePlugin()->getAsFlagLink($flag, $entity)['#action'];
  716. // if ($action) {
  717. $url = $link->getUrl()->toString();
  718. return '<div class="flag flag-' . $flagId . ' flag-' . $flagId . '-' . $entity->id() . ' action-' . $action . '"><a href="' . $url . '" class="use-ajax" rel="nofollow"></a></div>';
  719. // } else {
  720. // return null;
  721. // }
  722. }
  723. public static function flagAjaxBuilder($flagId, $entity, FlagService $flagService) {
  724. $flag = $flagService->getFlagById($flagId);
  725. $key = 'flag_' . $flag->id();
  726. $data = [
  727. '#lazy_builder' => ['flag.link_builder:build', [
  728. $entity->getEntityTypeId(),
  729. $entity->id(),
  730. $flag->id(),
  731. ]],
  732. '#create_placeholder' => TRUE,
  733. ];
  734. return [$key => $data];
  735. }
  736. /**
  737. * Updates the friendship status of these two users
  738. *
  739. * @param $uid
  740. * @param $uid_target
  741. * @param $unixtime
  742. * @param $friendStatus
  743. * @return \Drupal\Core\Database\StatementInterface|int|null
  744. */
  745. public static function updateFriendship($uid, $uid_target, $unixtime, $friendStatus) {
  746. // $query = Database::getConnection()->upsert('heartbeat_friendship')
  747. // ->fields(array(
  748. // 'uid' => $uid,
  749. // 'uid_target' => $uid_target,
  750. // 'created' => $unixtime,
  751. // 'status' => $friendStatus,
  752. // ))
  753. // ->key('uid_relation');
  754. // return $query->execute();
  755. $update = Database::getConnection()->update('heartbeat_friendship')
  756. ->fields(['status' => $friendStatus])
  757. ->condition('uid', $uid, '=')
  758. ->condition('uid_target', $uid_target, '=');
  759. if (!$update->execute()) {
  760. $insert = Database::getConnection()->insert('heartbeat_friendship')
  761. ->fields([
  762. 'uid' => $uid,
  763. 'uid_target' => $uid_target,
  764. 'created' => $unixtime,
  765. 'status' => $friendStatus
  766. ]);
  767. if (!$insert->execute()) {
  768. \Drupal::logger('Heartbeat')->error('Unable to update friendship between %uid and %uid_target', array('%uid' => $uid, '%uid_target' => $uid_target));
  769. }
  770. }
  771. if ($friendStatus === 1) {
  772. $update2 = Database::getConnection()->update('heartbeat_friendship')
  773. ->fields(['status' => $friendStatus])
  774. ->condition('uid', $uid_target, '=')
  775. ->condition('uid_target', $uid, '=');
  776. if (!$update2->execute()) {
  777. $insert2 = Database::getConnection()->insert('heartbeat_friendship')
  778. ->fields([
  779. 'uid' => $uid_target,
  780. 'uid_target' => $uid,
  781. 'created' => $unixtime,
  782. 'status' => $friendStatus
  783. ]);
  784. if (!$insert2->execute()) {
  785. \Drupal::logger('Heartbeat')->error('Unable to update friendship between %uid and %uid_target', array('%uid' => $uid_target, '%uid_target' => $uid));
  786. }
  787. }
  788. }
  789. return null;
  790. }
  791. /**
  792. * @param $tid
  793. */
  794. public static function updateTermUsage($tid, $vid = null) {
  795. $update = Database::getConnection()->update('taxonomy_term__field_count')
  796. ->expression('field_count_value', 'field_count_value + 1')
  797. ->condition('entity_id', $tid);
  798. if (!$update->execute()) {
  799. $insert = Database::getConnection()->insert('taxonomy_term__field_count')
  800. ->fields([
  801. 'entity_id' => $tid,
  802. 'revision_id' => $tid,
  803. 'bundle' => $vid,
  804. 'langcode' => \Drupal::languageManager()->getDefaultLanguage()->getId(),
  805. 'delta' => 0,
  806. 'field_count_value' => 1]);
  807. if (!$insert->execute()) {
  808. \Drupal::logger('Heartbeat')
  809. ->error('Unable to update term counts for Term with ID = %tid', array('%tid' => $tid));
  810. } else {
  811. self::newTermUsage($tid);
  812. }
  813. } else {
  814. self::newTermUsage($tid);
  815. }
  816. }
  817. /**
  818. * @param $tid
  819. */
  820. public static function newTermUsage($tid) {
  821. $insert = Database::getConnection()->insert('heartbeat_term_usage')
  822. ->fields(['tid' => $tid, 'timestamp' => time()]);
  823. if (!$insert->execute()) {
  824. \Drupal::logger('Heartbeat')->error('Unable to update term usage for Term with ID = %tid', array('%tid' => $tid));
  825. }
  826. }
  827. /**
  828. * Gets the Heartbeat user.
  829. *
  830. * @return int
  831. * The uid of the Heartbeat's user.
  832. */
  833. public function getUid() {
  834. return $this->get('uid');
  835. }
  836. /**
  837. * Sets the Heartbeat user.
  838. *
  839. * @param int uid
  840. * The Heartbeat user.
  841. *
  842. * @throws \Drupal\Core\Entity\EntityStorageException
  843. */
  844. public function setUid($uid)
  845. {
  846. $this->set('uid', $uid)->save();
  847. }
  848. /**
  849. * Gets the Heartbeat's associated node nid.
  850. *
  851. * @return int
  852. * The nid of the Heartbeat's associated node.
  853. */
  854. public function getNid() {
  855. return $this->get('nid');
  856. }
  857. /**
  858. * Sets the Heartbeat user.
  859. *
  860. * @param int uid
  861. * The Heartbeat user.
  862. *
  863. */
  864. public function setNid($nid) {
  865. $this->set('nid', $nid);
  866. }
  867. }