Browse Source

added some modules, updated some others

bach 2 years ago
parent
commit
072ea7de50
28 changed files with 2459 additions and 49 deletions
  1. 31 0
      sites/all/modules/contrib/admin/token/README.txt
  2. 3 4
      sites/all/modules/contrib/admin/token/tests/token_test.info
  3. 13 0
      sites/all/modules/contrib/admin/token/tests/token_test.tokens.inc
  4. 4 4
      sites/all/modules/contrib/admin/token/token.info
  5. 21 19
      sites/all/modules/contrib/admin/token/token.install
  6. 30 16
      sites/all/modules/contrib/admin/token/token.module
  7. 8 2
      sites/all/modules/contrib/admin/token/token.pages.inc
  8. 6 1
      sites/all/modules/contrib/admin/token/token.test
  9. 2 3
      sites/all/modules/contrib/admin/token/token.tokens.inc
  10. 339 0
      sites/all/modules/contrib/files/image_url_formatter/LICENSE.txt
  11. 16 0
      sites/all/modules/contrib/files/image_url_formatter/README.txt
  12. 12 0
      sites/all/modules/contrib/files/image_url_formatter/image_url_formatter.info
  13. 277 0
      sites/all/modules/contrib/files/image_url_formatter/image_url_formatter.module
  14. 339 0
      sites/all/modules/contrib/localisation/pathauto_i18n/LICENSE.txt
  15. 12 0
      sites/all/modules/contrib/localisation/pathauto_i18n/README.txt
  16. 12 0
      sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_node/pathauto_i18n_node.info
  17. 180 0
      sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_node/pathauto_i18n_node.module
  18. 12 0
      sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_taxonomy/pathauto_i18n_taxonomy.info
  19. 214 0
      sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_taxonomy/pathauto_i18n_taxonomy.module
  20. 11 0
      sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_user/pathauto_i18n_user.info
  21. 151 0
      sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_user/pathauto_i18n_user.module
  22. 16 0
      sites/all/modules/contrib/localisation/pathauto_i18n/pathauto_i18n.info
  23. 46 0
      sites/all/modules/contrib/localisation/pathauto_i18n/pathauto_i18n.install
  24. 251 0
      sites/all/modules/contrib/localisation/pathauto_i18n/pathauto_i18n.module
  25. 97 0
      sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n.test.inc
  26. 151 0
      sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n_node.test
  27. 116 0
      sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n_taxonomy.test
  28. 89 0
      sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n_user.test

+ 31 - 0
sites/all/modules/contrib/admin/token/README.txt

@@ -1,2 +1,33 @@
+INTRODUCTION
+------------
 
 Provides common and resuable token UI elements and missing core tokens.
+
+ * For a full description of the module, visit the project page:
+   https://drupal.org/project/token
+
+ * To submit bug reports and feature suggestions, or to track changes:
+   https://drupal.org/project/issues/token
+
+
+INSTALLATION
+------------
+
+Install as usual, see
+https://drupal.org/documentation/install/modules-themes/modules-7 for further
+information.
+
+
+TROUBLESHOOTING
+---------------
+
+Token module doesn't provide any visible functions to the user on its own, it
+just provides token handling services for other modules.
+
+
+MAINTAINERS
+-----------
+
+Current maintainers:
+
+ * Dave Reid (https://drupal.org/user/53892)

+ 3 - 4
sites/all/modules/contrib/admin/token/tests/token_test.info

@@ -5,9 +5,8 @@ core = 7.x
 files[] = token_test.module
 hidden = TRUE
 
-; Information added by drupal.org packaging script on 2013-02-24
-version = "7.x-1.5"
+; Information added by Drupal.org packaging script on 2021-03-02
+version = "7.x-1.8"
 core = "7.x"
 project = "token"
-datestamp = "1361665026"
-
+datestamp = "1614719852"

+ 13 - 0
sites/all/modules/contrib/admin/token/tests/token_test.tokens.inc

@@ -0,0 +1,13 @@
+<?php
+
+/**
+ * Implements hook_token_info()
+ */
+function token_test_token_info() {
+  $info['tokens']['node']['colons:in:name'] = array(
+    'name' => t('A test token with colons in the name'),
+    'description' => NULL,
+  );
+
+  return $info;
+}

+ 4 - 4
sites/all/modules/contrib/admin/token/token.info

@@ -1,11 +1,11 @@
 name = Token
 description = Provides a user interface for the Token API and some missing core tokens.
 core = 7.x
+dependencies[] = system (>=7.23)
 files[] = token.test
 
-; Information added by drupal.org packaging script on 2013-02-24
-version = "7.x-1.5"
+; Information added by Drupal.org packaging script on 2021-03-02
+version = "7.x-1.8"
 core = "7.x"
 project = "token"
-datestamp = "1361665026"
-
+datestamp = "1614719852"

+ 21 - 19
sites/all/modules/contrib/admin/token/token.install

@@ -21,18 +21,12 @@ function token_requirements($phase = 'runtime') {
         $problems = array_unique($problem['problems']);
         $problems = array_map('check_plain', $problems);
         $token_problems[$problem_key] = $problem['label'] . theme('item_list', array('items' => $problems));
+        $requirements['token-' . $problem_key] = array(
+          'title' => $problem['label'],
+          'value' => theme('item_list', array('items' => $problems)),
+          'severity' => $problem['severity'],
+        );
       }
-      else {
-        unset($token_problems[$problem_key]);
-      }
-    }
-    if (!empty($token_problems)) {
-      $requirements['token_problems'] = array(
-        'title' => $t('Tokens'),
-        'value' => $t('Problems detected'),
-        'severity' => REQUIREMENT_WARNING,
-        'description' => '<p>' . implode('</p><p>', $token_problems) . '</p>', //theme('item_list', array('items' => $token_problems)),
-      );
     }
   }
 
@@ -272,19 +266,24 @@ function token_get_token_problems() {
   $token_info = token_info();
   $token_problems = array(
     'not-array' => array(
-      'label' => t('The following tokens or token types are not defined as arrays:'),
+      'label' => t('Tokens or token types not defined as arrays'),
+      'severity' => REQUIREMENT_ERROR,
     ),
     'missing-info' => array(
-      'label' => t('The following tokens or token types are missing required name and/or description information:'),
+      'label' => t('Tokens or token types missing name property'),
+      'severity' => REQUIREMENT_WARNING,
     ),
     'type-no-tokens' => array(
-      'label' => t('The following token types do not have any tokens defined:'),
+      'label' => t('Token types do not have any tokens defined'),
+      'severity' => REQUIREMENT_INFO,
     ),
     'tokens-no-type' => array(
-      'label' => t('The following token types are not defined but have tokens:'),
+      'label' => t('Token types are not defined but have tokens'),
+      'severity' => REQUIREMENT_INFO,
     ),
     'duplicate' => array(
-      'label' => t('The following token or token types are defined by multiple modules:')
+      'label' => t('Token or token types are defined by multiple modules'),
+      'severity' => REQUIREMENT_ERROR,
     ),
   );
 
@@ -295,9 +294,12 @@ function token_get_token_problems() {
       $token_problems['not-array']['problems'][] = "\$info['types']['$type']";
       continue;
     }
-    elseif (!isset($type_info['name']) || !isset($type_info['description'])) {
+    elseif (!isset($type_info['name'])) {
       $token_problems['missing-info']['problems'][] = "\$info['types']['$type']";
     }
+    elseif (is_array($type_info['name'])) {
+      $token_problems['duplicate']['problems'][] = "\$info['types']['$type']";
+    }
     elseif (empty($token_info['tokens'][$real_type])) {
       $token_problems['type-no-tokens']['problems'][] = "\$info['tokens']['$real_type']";
     }
@@ -315,10 +317,10 @@ function token_get_token_problems() {
           $token_problems['not-array']['problems'][] = "\$info['tokens']['$type']['$token']";
           continue;
         }
-        elseif (!isset($tokens[$token]['name']) || !isset($tokens[$token]['description'])) {
+        elseif (!isset($tokens[$token]['name'])) {
           $token_problems['missing-info']['problems'][] = "\$info['tokens']['$type']['$token']";
         }
-        elseif (is_array($tokens[$token]['name']) || is_array($tokens[$token]['description'])) {
+        elseif (is_array($tokens[$token]['name'])) {
           $token_problems['duplicate']['problems'][] = "\$info['tokens']['$type']['$token']";
         }
       }

+ 30 - 16
sites/all/modules/contrib/admin/token/token.module

@@ -161,6 +161,8 @@ function token_theme() {
       'attributes' => array(),
       'empty' => '',
       'caption' => '',
+      'colgroups' => array(),
+      'sticky' => TRUE,
     ),
     'file' => 'token.pages.inc',
   );
@@ -180,7 +182,7 @@ function token_theme() {
       'text' => NULL,
       'options' => array(),
       'dialog' => TRUE,
-    ),
+    ) + $info['token_tree']['variables'],
     'file' => 'token.pages.inc',
   );
 
@@ -273,8 +275,12 @@ function token_form_block_admin_configure_alter(&$form, $form_state) {
  * Implements hook_widget_form_alter().
  */
 function token_field_widget_form_alter(&$element, &$form_state, $context) {
-  if (!empty($element['#description']) && is_string($element['#description'])) {
-    $element['#description'] = filter_xss_admin(token_replace($element['#description']));
+  if (!empty($element['#description']) && !empty($context['instance']['description'])) {
+    $instance = $context['instance'];
+    if (module_exists('i18n_field')) {
+      $instance = i18n_string_object_translate('field_instance', $instance);
+    }
+    $element['#description'] = field_filter_xss(token_replace($instance['description']));
   }
 }
 
@@ -412,12 +418,18 @@ function token_get_entity_mapping($value_type = 'token', $value = NULL, $fallbac
  */
 function token_entity_info_alter(&$info) {
   foreach (array_keys($info) as $entity_type) {
-    // Add a token view mode if it does not already exist.
-    if (!empty($info[$entity_type]['view modes']) && !isset($info[$entity_type]['view modes']['token'])) {
-      $info[$entity_type]['view modes']['token'] = array(
-        'label' => t('Tokens'),
-        'custom settings' => FALSE,
-      );
+    // Add a token view mode if it does not already exist. Only work with
+    // fieldable entities.
+    if (!empty($info[$entity_type]['fieldable'])) {
+      if (!isset($info[$entity_type])) {
+        $info[$entity_type]['view modes'] = array();
+      }
+      if (!isset($info[$entity_type]['view modes']['token'])) {
+        $info[$entity_type]['view modes']['token'] = array(
+          'label' => t('Tokens'),
+          'custom settings' => FALSE,
+        );
+      }
     }
 
     if (!empty($info[$entity_type]['token type'])) {
@@ -647,6 +659,10 @@ function token_get_invalid_tokens($type, $tokens) {
   $invalid_tokens = array();
 
   foreach ($tokens as $token => $full_token) {
+    if (isset($token_info['tokens'][$type][$token])) {
+      continue;
+    }
+
     // Split token up if it has chains.
     $parts = explode(':', $token, 2);
 
@@ -709,15 +725,13 @@ function token_element_validate(&$element, &$form_state) {
 
   // Validate if an element must have a minimum number of tokens.
   if (isset($element['#min_tokens']) && count($tokens) < $element['#min_tokens']) {
-    // @todo Change this error message to include the minimum number.
-    $error = format_plural($element['#min_tokens'], 'The %element-title cannot contain fewer than one token.', 'The %element-title must contain at least @count tokens.', array('%element-title' => $title));
+    $error = format_plural($element['#min_tokens'], '%name must contain at least one token.', '%name must contain at least @count tokens.', array('%name' => $title));
     form_error($element, $error);
   }
 
   // Validate if an element must have a maximum number of tokens.
   if (isset($element['#max_tokens']) && count($tokens) > $element['#max_tokens']) {
-    // @todo Change this error message to include the maximum number.
-    $error = format_plural($element['#max_tokens'], 'The %element-title must contain as most one token.', 'The %element-title must contain at most @count tokens.', array('%element-title' => $title));
+    $error = format_plural($element['#max_tokens'], '%name must contain at most one token.', '%name must contain at most @count tokens.', array('%name' => $title));
     form_error($element, $error);
   }
 
@@ -725,7 +739,7 @@ function token_element_validate(&$element, &$form_state) {
   if (isset($element['#token_types'])) {
     $invalid_tokens = token_get_invalid_tokens_by_context($tokens, $element['#token_types']);
     if ($invalid_tokens) {
-      form_error($element, t('The %element-title is using the following invalid tokens: @invalid-tokens.', array('%element-title' => $title, '@invalid-tokens' => implode(', ', $invalid_tokens))));
+      form_error($element, t('%name is using the following invalid tokens: @invalid-tokens.', array('%name' => $title, '@invalid-tokens' => implode(', ', $invalid_tokens))));
     }
   }
 
@@ -973,7 +987,7 @@ function _token_build_tree($token_type, array $options) {
       // parent.
       $token_parents[] = $token_type;
     }
-    elseif (in_array($token, array_slice($token_parents, 1))) {
+    elseif (in_array($token, array_slice($token_parents, 1), TRUE)) {
       // Prevent duplicate recursive tokens. For example, this will prevent
       // the tree from generating the following tokens or deeper:
       // [comment:parent:parent]
@@ -1216,7 +1230,7 @@ function token_menu_link_load_all_parents($mlid) {
   if (!isset($cache[$mlid])) {
     $cache[$mlid] = array();
     $plid = db_query("SELECT plid FROM {menu_links} WHERE mlid = :mlid", array(':mlid' => $mlid))->fetchField();
-    while ($plid && $parent = token_menu_link_load($plid)) {
+    while ($plid && $plid != $mlid && $parent = token_menu_link_load($plid)) {
       $cache[$mlid] = array($plid => $parent['title']) + $cache[$mlid];
       $plid = $parent['plid'];
     }

+ 8 - 2
sites/all/modules/contrib/admin/token/token.pages.inc

@@ -19,7 +19,12 @@ function theme_token_tree_link($variables) {
   }
 
   $info = token_theme();
-  $variables['options']['query']['options'] = array_intersect_key($variables, $info['token_tree']['variables']);
+  $tree_variables = array_intersect_key($variables, $info['token_tree']['variables']);
+  $tree_variables = drupal_array_diff_assoc_recursive($tree_variables, $info['token_tree']['variables']);
+  if (!isset($variables['options']['query']['options'])) {
+    $variables['options']['query']['options'] = array();
+  }
+  $variables['options']['query']['options'] += $tree_variables;
 
   // We should never pass the dialog option to theme_token_tree(). It is only
   // used for this function.
@@ -194,7 +199,7 @@ function _token_token_tree_format_row($token, array $token_info, $is_group = FAL
   $row = $defaults;
   $row['id'] = _token_clean_css_identifier($token);
   $row['data']['name'] = $token_info['name'];
-  $row['data']['description'] = $token_info['description'];
+  $row['data']['description'] = isset($token_info['description']) ? $token_info['description'] : '';
 
   if ($is_group) {
     // This is a token type/group.
@@ -202,6 +207,7 @@ function _token_token_tree_format_row($token, array $token_info, $is_group = FAL
   }
   else {
     // This is a token.
+    $row['data']['token'] = array();
     $row['data']['token']['data'] = $token;
     $row['data']['token']['class'][] = 'token-key';
     if (isset($token_info['value'])) {

+ 6 - 1
sites/all/modules/contrib/admin/token/token.test

@@ -137,6 +137,7 @@ class TokenUnitTestCase extends TokenTestHelper {
         '[node:created:short]',
         '[node:created:custom:invalid]',
         '[node:created:custom:mm-YYYY]',
+        '[node:colons:in:name]',
         '[site:name]',
         '[site:slogan]',
         '[current-date:short]',
@@ -147,6 +148,7 @@ class TokenUnitTestCase extends TokenTestHelper {
         '[node:title:invalid]',
         '[node:created:invalid]',
         '[node:created:short:invalid]',
+        '[node:colons:in:name:invalid]',
         '[invalid:title]',
         '[site:invalid]',
         '[user:ip-address]',
@@ -166,6 +168,7 @@ class TokenUnitTestCase extends TokenTestHelper {
         '[node:created:short]',
         '[node:created:custom:invalid]',
         '[node:created:custom:mm-YYYY]',
+        '[node:colons:in:name]',
         '[site:name]',
         '[site:slogan]',
         '[user:uid]',
@@ -176,6 +179,7 @@ class TokenUnitTestCase extends TokenTestHelper {
         '[node:title:invalid]',
         '[node:created:invalid]',
         '[node:created:short:invalid]',
+        '[node:colons:in:name:invalid]',
         '[invalid:title]',
         '[site:invalid]',
         '[user:ip-address]',
@@ -952,6 +956,7 @@ class TokenArrayTestCase extends TokenTestHelper {
       'last' => 'b',
       'value:0' => 'a',
       'value:2' => 'c',
+      'value:#property' => NULL,
       'count' => 4,
       'keys' => '2, 0, 4, 1',
       'keys:value:3' => '1',
@@ -1076,7 +1081,7 @@ class TokenBlockTestCase extends TokenTestHelper {
     $edit['body[value]'] = 'This is the test token title block.';
     $this->drupalPost('admin/structure/block/add', $edit, 'Save block');
     // Ensure token validation is working on the block.
-    $this->assertText('The Block title is using the following invalid tokens: [user:name].');
+    $this->assertText('Block title is using the following invalid tokens: [user:name].');
 
     // Create the block for real now with a valid title.
     $edit['title'] = '[current-page:title] block title';

+ 2 - 3
sites/all/modules/contrib/admin/token/token.tokens.inc

@@ -848,7 +848,7 @@ function token_tokens($type, $tokens, array $data = array(), array $options = ar
     // [array:value:*] dynamic tokens.
     if ($value_tokens = token_find_with_prefix($tokens, 'value')) {
       foreach ($value_tokens as $key => $original) {
-        if ($key[0] !== '#' && isset($array[$key])) {
+        if (array_key_exists($key, $array) && in_array($key, $keys)) {
           $replacements[$original] = token_render_array_value($array[$key], $options);
         }
       }
@@ -1392,7 +1392,6 @@ function field_token_info_alter(&$info) {
  */
 function field_tokens($type, $tokens, array $data = array(), array $options = array()) {
   $replacements = array();
-  $sanitize = !empty($options['sanitize']);
   $langcode = isset($options['language']) ? $options['language']->language : NULL;
 
   // Entity tokens.
@@ -1437,7 +1436,7 @@ function field_tokens($type, $tokens, array $data = array(), array $options = ar
 /**
  * Pre-render callback for field output used with tokens.
  */
-function token_pre_render_field_token(&$elements) {
+function token_pre_render_field_token($elements) {
   // Remove the field theme hook, attachments, and JavaScript states.
   unset($elements['#theme']);
   unset($elements['#states']);

+ 339 - 0
sites/all/modules/contrib/files/image_url_formatter/LICENSE.txt

@@ -0,0 +1,339 @@
+                    GNU GENERAL PUBLIC LICENSE
+                       Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                            Preamble
+
+  The licenses for most software are designed to take away your
+freedom to share and change it.  By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users.  This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it.  (Some other Free Software Foundation software is covered by
+the GNU Lesser General Public License instead.)  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+  To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have.  You must make sure that they, too, receive or can get the
+source code.  And you must show them these terms so they know their
+rights.
+
+  We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+  Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software.  If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+  Finally, any free program is threatened constantly by software
+patents.  We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary.  To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+                    GNU GENERAL PUBLIC LICENSE
+   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+  0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License.  The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language.  (Hereinafter, translation is included without limitation in
+the term "modification".)  Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope.  The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+  1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+  2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+    a) You must cause the modified files to carry prominent notices
+    stating that you changed the files and the date of any change.
+
+    b) You must cause any work that you distribute or publish, that in
+    whole or in part contains or is derived from the Program or any
+    part thereof, to be licensed as a whole at no charge to all third
+    parties under the terms of this License.
+
+    c) If the modified program normally reads commands interactively
+    when run, you must cause it, when started running for such
+    interactive use in the most ordinary way, to print or display an
+    announcement including an appropriate copyright notice and a
+    notice that there is no warranty (or else, saying that you provide
+    a warranty) and that users may redistribute the program under
+    these conditions, and telling the user how to view a copy of this
+    License.  (Exception: if the Program itself is interactive but
+    does not normally print such an announcement, your work based on
+    the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole.  If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works.  But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+  3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+    a) Accompany it with the complete corresponding machine-readable
+    source code, which must be distributed under the terms of Sections
+    1 and 2 above on a medium customarily used for software interchange; or,
+
+    b) Accompany it with a written offer, valid for at least three
+    years, to give any third party, for a charge no more than your
+    cost of physically performing source distribution, a complete
+    machine-readable copy of the corresponding source code, to be
+    distributed under the terms of Sections 1 and 2 above on a medium
+    customarily used for software interchange; or,
+
+    c) Accompany it with the information you received as to the offer
+    to distribute corresponding source code.  (This alternative is
+    allowed only for noncommercial distribution and only if you
+    received the program in object code or executable form with such
+    an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it.  For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable.  However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+  4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License.  Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+  5. You are not required to accept this License, since you have not
+signed it.  However, nothing else grants you permission to modify or
+distribute the Program or its derivative works.  These actions are
+prohibited by law if you do not accept this License.  Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+  6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions.  You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+  7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all.  For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices.  Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+  8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded.  In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+  9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number.  If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation.  If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+  10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission.  For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this.  Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+                            NO WARRANTY
+
+  11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.  EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.  THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU.  SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+  12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+                     END OF TERMS AND CONDITIONS
+
+            How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    This program is free software; you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation; either version 2 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License along
+    with this program; if not, write to the Free Software Foundation, Inc.,
+    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+    Gnomovision version 69, Copyright (C) year name of author
+    Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary.  Here is a sample; alter the names:
+
+  Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+  `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+  <signature of Ty Coon>, 1 April 1989
+  Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs.  If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library.  If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.

+ 16 - 0
sites/all/modules/contrib/files/image_url_formatter/README.txt

@@ -0,0 +1,16 @@
+This module add a url formatter for image field. 
+Then you can output image url directly.
+
+Most of the code, maybe more 90%, is just copy from the drupal core.
+I think it is stable enough.
+
+Usage
+(1)After install this module,then you add an image field,
+then you can goto this bundle's manage display page,here at format column,
+you can choose "Image URL" instead of "Image".
+
+(2)When you add an iamge field in your views,
+then you config the Formatter for this field,
+ here you can choose "Image URL" instead of "Image".
+
+Then you can output image's URL instead of Image itself.

+ 12 - 0
sites/all/modules/contrib/files/image_url_formatter/image_url_formatter.info

@@ -0,0 +1,12 @@
+name = Image URL Formatter
+description = "Add an URL formatter for image field"
+core = 7.x
+dependencies[] = field
+dependencies[] = image
+files[] = image_url_formatter.module
+; Information added by drupal.org packaging script on 2013-09-10
+version = "7.x-1.4"
+core = "7.x"
+project = "image_url_formatter"
+datestamp = "1378820223"
+

+ 277 - 0
sites/all/modules/contrib/files/image_url_formatter/image_url_formatter.module

@@ -0,0 +1,277 @@
+<?php
+
+/**
+ * @file
+ * Add an URL formatter for image field
+ */
+
+/**
+ * Define constants for determine which type of URL should be used.
+ */
+define('IMAGE_URL_FORMATTER_URI_PATH', '3');
+define('IMAGE_URL_FORMATTER_RELATIVE_PATH', '2');
+define('IMAGE_URL_FORMATTER_ABSOLUTE_PATH', '1');
+define('IMAGE_URL_FORMATTER_FULL_URL', '0');
+
+/**
+ * Implements hook_theme().
+ */
+function image_url_formatter_theme() {
+  return array(
+    'image_url_formatter' => array(
+      'variables' => array(
+        'item' => NULL,
+        'path' => NULL,
+        'image_style' => NULL,
+        'url_type' => NULL,
+      ),
+    ),
+  );
+}
+
+/**
+ * Implements hook_field_formatter_info().
+ */
+function image_url_formatter_field_formatter_info() {
+  $formatters = array(
+    'image_url' => array(
+      'label' => t('Image URL'),
+      'field types' => array('image', 'imagefield_crop'),
+      'settings' => array(
+        'url_type' => '',
+        'image_style' => '',
+        'image_link' => '',
+      ),
+    ),
+  );
+
+  return $formatters;
+}
+
+/**
+ * Implements hook_field_formatter_settings_form().
+ */
+function image_url_formatter_field_formatter_settings_form($field, $instance, $view_mode, $form, &$form_state) {
+  $display = $instance['display'][$view_mode];
+  $settings = $display['settings'];
+
+  $element['url_type'] = array(
+    '#title' => t('URL type'),
+    '#type' => 'radios',
+    '#options' => array(
+      IMAGE_URL_FORMATTER_URI_PATH => t('URI path'),
+      IMAGE_URL_FORMATTER_RELATIVE_PATH => t('Relative file path'),
+      IMAGE_URL_FORMATTER_ABSOLUTE_PATH => t('Absolute file path (recommended)'),
+      IMAGE_URL_FORMATTER_FULL_URL => t('Full URL'),
+    ),
+    '#default_value' => $settings['url_type'],
+  );
+  $element['url_type'][IMAGE_URL_FORMATTER_URI_PATH]['#description'] = t("Uses the URI path, like: 'public://image.png'");
+  $element['url_type'][IMAGE_URL_FORMATTER_RELATIVE_PATH]['#description'] = t("No base URL or leading slash, like: 'sites/default/files/image.png'");
+  $element['url_type'][IMAGE_URL_FORMATTER_ABSOLUTE_PATH]['#description'] = t("With leading slash, no base URL, like: '/sites/default/files/image.png'");
+  $element['url_type'][IMAGE_URL_FORMATTER_FULL_URL]['#description'] = t("Like: 'http://example.com/sites/default/files/image.png'");
+
+  $image_styles = image_style_options(FALSE);
+  $element['image_style'] = array(
+    '#title' => t('Image style'),
+    '#type' => 'select',
+    '#default_value' => $settings['image_style'],
+    '#empty_option' => t('None (original image)'),
+    '#options' => $image_styles,
+  );
+
+  $link_types = array(
+    'content' => t('Content'),
+    'file' => t('File'),
+  );
+  $element['image_link'] = array(
+    '#title' => t('Link image url to'),
+    '#type' => 'select',
+    '#default_value' => $settings['image_link'],
+    '#empty_option' => t('Nothing'),
+    '#options' => $link_types,
+  );
+
+  return $element;
+}
+
+/**
+ * Implements hook_field_formatter_settings_summary().
+ */
+function image_url_formatter_field_formatter_settings_summary($field, $instance, $view_mode) {
+  $display = $instance['display'][$view_mode];
+  $settings = $display['settings'];
+
+  $summary = array();
+
+  switch ($settings['url_type']) {
+    case IMAGE_URL_FORMATTER_URI_PATH:
+      $summary[] = t('Use uri path');
+      break;
+
+    case IMAGE_URL_FORMATTER_RELATIVE_PATH:
+      $summary[] = t('Use relative path');
+      break;
+
+    case IMAGE_URL_FORMATTER_ABSOLUTE_PATH:
+      $summary[] = t('Use absolute path');
+      break;
+
+    case IMAGE_URL_FORMATTER_FULL_URL:
+      $summary[] = t('Use full URL');
+      break;
+  }
+
+  $image_styles = image_style_options(FALSE);
+  // Unset possible 'No defined styles' option.
+  unset($image_styles['']);
+  // Styles could be lost because of enabled/disabled modules that defines
+  // their styles in code.
+  if (isset($image_styles[$settings['image_style']])) {
+    $summary[] = t('URL for image style: @style', array('@style' => $image_styles[$settings['image_style']]));
+  }
+  else {
+    $summary[] = t('Original image URL');
+  }
+
+  $link_types = array(
+    'content' => t('Linked to content'),
+    'file' => t('Linked to file'),
+  );
+  // Display this setting only if image is linked.
+  if (isset($link_types[$settings['image_link']])) {
+    $summary[] = $link_types[$settings['image_link']];
+  }
+
+  return implode('<br />', $summary);
+}
+/**
+ * Implements hook_field_formatter_view().
+ */
+function image_url_formatter_field_formatter_view($entity_type, $entity, $field, $instance, $langcode, $items, $display) {
+  $element = array();
+
+  switch ($display['type']) {
+    case 'image_url':
+      // Check if the formatter involves a link.
+      if ($display['settings']['image_link'] == 'content') {
+        $uri = entity_uri($entity_type, $entity);
+      }
+      elseif ($display['settings']['image_link'] == 'file') {
+        $link_file = TRUE;
+      }
+
+      foreach ($items as $delta => $item) {
+        if (isset($link_file)) {
+          $uri = array(
+            'path' => file_create_url($item['uri']),
+            'options' => array(),
+          );
+        }
+        $element[$delta] = array(
+          '#theme' => 'image_url_formatter',
+          '#item' => $item,
+          '#image_style' => $display['settings']['image_style'],
+          '#path' => isset($uri) ? $uri : '',
+          '#url_type' => $display['settings']['url_type'],
+        );
+      }
+
+      break;
+
+  }
+
+  return $element;
+}
+
+/**
+ * Returns HTML for an image url field formatter.
+ *
+ * @param array $variables
+ *   An associative array containing:
+ *   - item: An array of image data.
+ *   - image_style: An optional image style.
+ *   - path: An array containing the link 'path' and link 'options'.
+ *
+ * @ingroup themeable
+ */
+function theme_image_url_formatter($variables) {
+  $item = $variables['item'];
+  $image = array(
+    'path' => $item['uri'],
+    'alt' => $item['alt'],
+  );
+  // Do not output an empty 'title' attribute.
+  if (drupal_strlen($item['title']) > 0) {
+    $image['title'] = $item['title'];
+  }
+  // Return the URI path.
+  if ($variables['url_type'] == 3) {
+    return $item['uri'];
+  }
+  $output = file_create_url($item['uri']);
+  if ($variables['image_style']) {
+    $image['style_name'] = $variables['image_style'];
+    $output = image_style_url($image['style_name'], $item['uri']);
+  }
+  $output = image_url_formatter_convert_full_url($output, $variables['url_type']);
+  if ($variables['path']) {
+    $path = $variables['path']['path'];
+    $path = image_url_formatter_convert_full_url($path, $variables['url_type']);
+    $options = $variables['path']['options'];
+    // When displaying an image inside a link, the html option must be TRUE.
+    $options['html'] = TRUE;
+    $output = l($output, $path, $options);
+  }
+
+  return $output;
+}
+
+/**
+ * Converts a full URL to the choosen format.
+ *
+ * @param string $url
+ *   The full URL to convet.
+ * @param constant $format
+ *   IMAGE_URL_FORMATTER_RELATIVE_PATH for relative path,
+ *   IMAGE_URL_FORMATTER_ABSOLUTE_PATH for absolute path,
+ *   IMAGE_URL_FORMATTER_FULL_URL for full URL.
+ *
+ * @return string
+ *   The converted URL.
+ */
+function image_url_formatter_convert_full_url($url, $format = IMAGE_URL_FORMATTER_FULL_URL) {
+  switch ($format) {
+    case IMAGE_URL_FORMATTER_RELATIVE_PATH:
+      $url = _image_url_formatter_get_relative_file_url($url);
+      break;
+
+    case IMAGE_URL_FORMATTER_ABSOLUTE_PATH:
+      $url = _image_url_formatter_get_absolute_file_url($url);
+      break;
+  }
+
+  return $url;
+}
+
+/**
+ * Returns an absolute url.
+ */
+function _image_url_formatter_get_absolute_file_url($url) {
+  global $base_url;
+  if (strpos($url, $base_url) === 0) {
+    $url = base_path() . ltrim(str_replace($GLOBALS['base_url'], '', $url), '/');
+  }
+  return $url;
+}
+
+/**
+ * Returns a relative url.
+ */
+function _image_url_formatter_get_relative_file_url($url) {
+  $url = _image_url_formatter_get_absolute_file_url($url);
+  if ($url[0] == '/') {
+    $url = substr($url, 1);
+  }
+  return $url;
+}

+ 339 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/LICENSE.txt

@@ -0,0 +1,339 @@
+                    GNU GENERAL PUBLIC LICENSE
+                       Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                            Preamble
+
+  The licenses for most software are designed to take away your
+freedom to share and change it.  By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users.  This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it.  (Some other Free Software Foundation software is covered by
+the GNU Lesser General Public License instead.)  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+  To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have.  You must make sure that they, too, receive or can get the
+source code.  And you must show them these terms so they know their
+rights.
+
+  We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+  Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software.  If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+  Finally, any free program is threatened constantly by software
+patents.  We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary.  To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+                    GNU GENERAL PUBLIC LICENSE
+   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+  0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License.  The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language.  (Hereinafter, translation is included without limitation in
+the term "modification".)  Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope.  The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+  1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+  2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+    a) You must cause the modified files to carry prominent notices
+    stating that you changed the files and the date of any change.
+
+    b) You must cause any work that you distribute or publish, that in
+    whole or in part contains or is derived from the Program or any
+    part thereof, to be licensed as a whole at no charge to all third
+    parties under the terms of this License.
+
+    c) If the modified program normally reads commands interactively
+    when run, you must cause it, when started running for such
+    interactive use in the most ordinary way, to print or display an
+    announcement including an appropriate copyright notice and a
+    notice that there is no warranty (or else, saying that you provide
+    a warranty) and that users may redistribute the program under
+    these conditions, and telling the user how to view a copy of this
+    License.  (Exception: if the Program itself is interactive but
+    does not normally print such an announcement, your work based on
+    the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole.  If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works.  But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+  3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+    a) Accompany it with the complete corresponding machine-readable
+    source code, which must be distributed under the terms of Sections
+    1 and 2 above on a medium customarily used for software interchange; or,
+
+    b) Accompany it with a written offer, valid for at least three
+    years, to give any third party, for a charge no more than your
+    cost of physically performing source distribution, a complete
+    machine-readable copy of the corresponding source code, to be
+    distributed under the terms of Sections 1 and 2 above on a medium
+    customarily used for software interchange; or,
+
+    c) Accompany it with the information you received as to the offer
+    to distribute corresponding source code.  (This alternative is
+    allowed only for noncommercial distribution and only if you
+    received the program in object code or executable form with such
+    an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it.  For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable.  However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+  4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License.  Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+  5. You are not required to accept this License, since you have not
+signed it.  However, nothing else grants you permission to modify or
+distribute the Program or its derivative works.  These actions are
+prohibited by law if you do not accept this License.  Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+  6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions.  You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+  7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all.  For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices.  Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+  8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded.  In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+  9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number.  If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation.  If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+  10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission.  For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this.  Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+                            NO WARRANTY
+
+  11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.  EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.  THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU.  SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+  12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+                     END OF TERMS AND CONDITIONS
+
+            How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    This program is free software; you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation; either version 2 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License along
+    with this program; if not, write to the Free Software Foundation, Inc.,
+    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+    Gnomovision version 69, Copyright (C) year name of author
+    Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary.  Here is a sample; alter the names:
+
+  Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+  `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+  <signature of Ty Coon>, 1 April 1989
+  Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs.  If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library.  If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.

+ 12 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/README.txt

@@ -0,0 +1,12 @@
+README.txt
+==========
+
+This module provides functionality to create aliases for each language.
+
+USAGE:
+==========
+Set up and use:
+1) Install and activate the module.
+2) Enable multilingual support for nodes(needs only for nodes)
+3) Configure alias patterns on page admin/config/search/path/patterns
+4) Create entity with option "Generate automatic URL alias for other languages"

+ 12 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_node/pathauto_i18n_node.info

@@ -0,0 +1,12 @@
+name = Pathauto i18n node
+description = Provides the ability to add aliases for multilanguage nodes
+core = 7.x
+dependencies[] = node
+dependencies[] = pathauto_i18n
+
+; Information added by Drupal.org packaging script on 2016-03-02
+version = "7.x-1.5"
+core = "7.x"
+project = "pathauto_i18n"
+datestamp = "1456925640"
+

+ 180 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_node/pathauto_i18n_node.module

@@ -0,0 +1,180 @@
+<?php
+/**
+ * @file
+ * Provides tools for creating multilanguage aliases for nodes.
+ */
+
+/**
+ * Implements hook_form_BASE_FORM_ID_alter().
+ */
+function pathauto_i18n_form_node_form_alter(&$form, &$form_state) {
+  pathauto_i18n_configuration_form($form, $form_state['node'], 'node', $form['#node']->type);
+}
+
+/**
+ * Implements hook_node_insert().
+ */
+function pathauto_i18n_node_node_insert($node) {
+  pathauto_i18n_init_property($node, 'node', $node->type);
+  pathauto_i18n_process_entity_object($node, 'node', $node->path['pathauto_i18n_status'], 'insert');
+}
+
+/**
+ * Implements hook_node_update().
+ */
+function pathauto_i18n_node_node_update($node) {
+  pathauto_i18n_init_property($node, 'node', $node->type);
+  pathauto_i18n_process_entity_object($node, 'node', $node->path['pathauto_i18n_status'], 'update');
+}
+
+/**
+ * Implements hook_node_delete().
+ */
+function pathauto_i18n_node_node_delete($node) {
+  pathauto_i18n_delete_settings($node->nid, 'node');
+}
+
+/**
+ * Implements hook_node_load().
+ */
+function pathauto_i18n_node_node_load($nodes, $types) {
+  // Attach pathauto i18n settings to node object.
+  foreach ($nodes as $node) {
+    $nids[] = $node->nid;
+  }
+
+  if (!empty($nids)) {
+    $result = pathauto_i18n_load_settings($nids, 'node');
+    $settings = array();
+    foreach ($result as $record) {
+      $settings[$record->entity_id] = $record->path_status;
+    }
+    foreach ($nodes as $nid => $node) {
+      if (array_key_exists($nid, $settings)) {
+        $nodes[$nid]->path['pathauto_i18n_status'] = $settings[$nid];
+      }
+      else {
+        $nodes[$nid]->path['pathauto_i18n_status'] = pathauto_i18n_get_bundle_default('node', $node->type);
+      }
+    }
+  }
+}
+
+/**
+ * Implements hook_pathauto_alias_alter().
+ */
+function pathauto_i18n_node_pathauto_alias_alter(&$alias, &$context) {
+  $operations = array('insert', 'update', 'bulkupdate');
+  // Skip insert of alias for all languages.
+  if (!empty($context['module']) && $context['module'] == 'node' && in_array($context['op'], $operations) && !empty($context['data']['node'])) {
+    $entity = $context['data']['node'];
+
+    // Run bulk update.
+    if ($context['op'] == 'bulkupdate') {
+      pathauto_i18n_node_node_update($entity);
+    }
+
+    if (isset($entity->path['pathauto_i18n_status']) && $entity->path['pathauto_i18n_status'] && $context['language'] == LANGUAGE_NONE) {
+      $alias = '';
+    }
+  }
+}
+
+/**
+ * Implements hook_form_alter().
+ */
+function pathauto_i18n_node_form_alter(&$form, &$form_state, $form_id) {
+  if (!empty($form['#node_edit_form']) && $form['#node_edit_form']) {
+    // Add pathauto value if pathauto_i18n_status TRUE.
+    // Remove alias value to prevent overwriting.
+    if (isset($form['#node']->path['pathauto_i18n_status']) && $form['#node']->path['pathauto_i18n_status']) {
+      $form['path']['pathauto']['#default_value'] = TRUE;
+      $form['path']['alias']['#default_value'] = '';
+    }
+  }
+}
+
+/**
+ * Implements hook_module_implements_alter().
+ */
+function pathauto_i18n_node_module_implements_alter(&$implementations, $hook) {
+  // Move pathauto_i18n_node_form_alter to the end of the list.
+  // By default there not available pathauto value.
+  if ($hook == 'form_alter') {
+    $group = $implementations['pathauto_i18n_node'];
+    unset($implementations['pathauto_i18n_node']);
+    $implementations['pathauto_i18n_node'] = $group;
+  }
+}
+
+/**
+ * Implements hook_action_info().
+ */
+function pathauto_i18n_node_action_info() {
+  return array(
+    'pathauto_i18n_node_generate_alias' => array(
+      'type' => 'node',
+      'label' => t('Enable generation of aliases for all languages'),
+      'configurable' => FALSE,
+      'behavior' => array('changes_property'),
+      'triggers' => array(
+        'node_presave',
+      ),
+    ),
+    'pathauto_i18n_node_remove_alias' => array(
+      'type' => 'node',
+      'label' => t('Disable generation of aliases for all languages'),
+      'configurable' => FALSE,
+      'behavior' => array('changes_property'),
+      'triggers' => array(
+        'node_presave',
+      ),
+    ),
+  );
+}
+
+/**
+ * Sets the status of a pathauto_i18n_status to 1.
+ *
+ * @param object $node
+ *   A node object.
+ * @param array $context
+ *   (optional) Array of additional information about what triggered the action.
+ *   Not used for this action.
+ *
+ * @ingroup actions
+ */
+function pathauto_i18n_node_generate_alias($node, $context = array()) {
+  $node->path['pathauto_i18n_status'] = 1;
+}
+
+/**
+ * Sets the status of a pathauto_i18n_status to 0.
+ *
+ * @param object $node
+ *   A node object.
+ * @param array $context
+ *   (optional) Array of additional information about what triggered the action.
+ *   Not used for this action.
+ *
+ * @ingroup actions
+ */
+function pathauto_i18n_node_remove_alias($node, $context = array()) {
+  $node->path['pathauto_i18n_status'] = 0;
+}
+
+/**
+ * Implements hook_form_FORM_ID_alter().
+ *
+ * This will add pathauto_i18n options to the node type form. These settings
+ * will be used as default for every node of this node type.
+ */
+function pathauto_i18n_node_form_node_type_form_alter(&$form, $form_state) {
+  // Add the pathauto i18n form.
+  $form['workflow']['pathauto_i18n_default_node'] = array(
+    '#type' => 'checkbox',
+    '#title' => t('Generate automatic URL alias for all languages by default'),
+    '#default_value' => pathauto_i18n_get_bundle_default('node', $form['#node_type']->type),
+    '#description' => t('Set the default behaviour generating aliases for all available languages.'),
+  );
+}

+ 12 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_taxonomy/pathauto_i18n_taxonomy.info

@@ -0,0 +1,12 @@
+name = Pathauto i18n taxonomy
+description = Provides the ability to add aliases for multilanguage taxonomy
+core = 7.x
+dependencies[] = pathauto_i18n
+dependencies[] = taxonomy
+
+; Information added by Drupal.org packaging script on 2016-03-02
+version = "7.x-1.5"
+core = "7.x"
+project = "pathauto_i18n"
+datestamp = "1456925640"
+

+ 214 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_taxonomy/pathauto_i18n_taxonomy.module

@@ -0,0 +1,214 @@
+<?php
+/**
+ * @file
+ * Provides tools for creating multilanguage aliases for taxonomy terms.
+ */
+
+/**
+ * Implements hook_form_BASE_FORM_ID_alter().
+ */
+function pathauto_i18n_taxonomy_form_taxonomy_form_term_alter(&$form, &$form_state) {
+  // Exclude elements from deletion confirm form.
+  if ($form['#theme'] != 'confirm_form') {
+    pathauto_i18n_configuration_form($form, $form_state['term'], 'term', $form_state['term']->vocabulary_machine_name);
+  }
+}
+
+/**
+ * Implements hook_form_BASE_FORM_ID_alter().
+ */
+function pathauto_i18n_taxonomy_form_pathauto_patterns_form_alter(&$form, &$form_state) {
+  $languages = language_list();
+  $default_pattern_name = 'pathauto_taxonomy_term_pattern';
+  $default_pattern = $form['taxonomy_term'][$default_pattern_name];
+  // Remove parents handlers.
+  unset($default_pattern['#parents']);
+  $form['taxonomy_term']['token_help']['#weight'] = 1;
+  foreach (element_children($form['taxonomy_term']) as $term_pattern_name) {
+    if ($term_pattern_name != $default_pattern_name && $term_pattern_name != 'token_help') {
+      foreach ($languages as $language) {
+        $vocabulary = pathauto_i18n_taxonomy_get_vocabulary_name($term_pattern_name);
+        if ($vocabulary) {
+          $pattern_name = 'pathauto_taxonomy_term_' . $vocabulary . '_' . $language->language . '_pattern';
+          $form['taxonomy_term'][$pattern_name] = $default_pattern;
+          $form['taxonomy_term'][$pattern_name]['#title'] = t('Pattern for all @language @vocabulary paths', array('@language' => $language->name, '@vocabulary' => $vocabulary));
+          $form['taxonomy_term'][$pattern_name]['#default_value'] = variable_get($pattern_name, '');
+        }
+      }
+    }
+  }
+}
+
+/**
+ * Get vocabulary name from pattern name.
+ */
+function pathauto_i18n_taxonomy_get_vocabulary_name($pattern_name) {
+  $scope = str_replace(array('pathauto_taxonomy_term_', '_pattern'), array('', ''), $pattern_name);
+  return !empty($scope) ? $scope : FALSE;
+}
+
+/**
+ * Implements hook_taxonomy_term_insert().
+ */
+function pathauto_i18n_taxonomy_taxonomy_term_insert($term) {
+  pathauto_i18n_init_property($term, 'taxonomy', $term->vocabulary_machine_name);
+  pathauto_i18n_process_entity_object($term, 'taxonomy_term', $term->path['pathauto_i18n_status'], 'insert');
+}
+
+/**
+ * Implements hook_taxonomy_term_update().
+ */
+function pathauto_i18n_taxonomy_taxonomy_term_update($term) {
+  pathauto_i18n_init_property($term, 'taxonomy', $term->vocabulary_machine_name);
+  pathauto_i18n_process_entity_object($term, 'taxonomy_term', $term->path['pathauto_i18n_status'], 'update');
+}
+
+/**
+ * Implements hook_taxonomy_term_delete().
+ */
+function pathauto_i18n_taxonomy_taxonomy_term_delete($term) {
+  pathauto_i18n_delete_settings($term->tid, 'taxonomy_term');
+}
+
+/**
+ * Implements hook_taxonomy_term_load().
+ */
+function pathauto_i18n_taxonomy_taxonomy_term_load($terms) {
+  // Attach pathauto i18n settings to taxonomy_term object.
+  foreach ($terms as $term) {
+    $tids[] = $term->tid;
+  }
+
+  if (!empty($tids)) {
+    $result = pathauto_i18n_load_settings($tids, 'taxonomy_term');
+    $settings = array();
+    foreach ($result as $record) {
+      $settings[$record->entity_id] = $record->path_status;
+    }
+    foreach ($terms as $tid => $term) {
+      if (array_key_exists($tid, $settings)) {
+        $terms[$tid]->path['pathauto_i18n_status'] = $settings[$tid];
+      }
+      else {
+        $terms[$tid]->path['pathauto_i18n_status'] = 0;
+      }
+    }
+  }
+}
+
+/**
+ * Implements hook_pathauto_alias_alter().
+ */
+function pathauto_i18n_taxonomy_pathauto_alias_alter(&$alias, &$context) {
+  $operations = array('insert', 'update', 'bulkupdate');
+  // Skip insert of alias for all languages.
+  if (!empty($context['module']) && ($context['module'] == 'term' || $context['module'] == 'taxonomy_term') && in_array($context['op'], $operations)) {
+    $entity = FALSE;
+
+    // On creating taxonomy term has term key on updating taxonomy_term.
+    if (!empty($context['data']['term'])) {
+      $entity = $context['data']['term'];
+    }
+    if (!empty($context['data']['taxonomy_term'])) {
+      $entity = $context['data']['taxonomy_term'];
+    }
+
+    // Run bulk update.
+    if ($context['op'] == 'bulkupdate') {
+      pathauto_i18n_taxonomy_taxonomy_term_update($entity);
+    }
+
+    if ($entity && isset($entity->path['pathauto_i18n_status']) && $entity->path['pathauto_i18n_status'] && $context['language'] == LANGUAGE_NONE) {
+      $alias = '';
+    }
+  }
+}
+
+/**
+ * Implements hook_form_alter().
+ */
+function pathauto_i18n_taxonomy_form_alter(&$form, &$form_state, $form_id) {
+  if ($form_id == 'taxonomy_form_term') {
+    // Add pathauto value if pathauto_i18n_status TRUE.
+    // Remove alias value to prevent overwriting.
+    $term = FALSE;
+    if (isset($form['#term'])) {
+      $term = (object) $form['#term'];
+    }
+    if ($term && isset($term->path['pathauto_i18n_status']) && $term->path['pathauto_i18n_status']) {
+      $form['path']['pathauto']['#default_value'] = TRUE;
+      $form['path']['alias']['#default_value'] = '';
+    }
+  }
+}
+
+/**
+ * Implements hook_module_implements_alter().
+ */
+function pathauto_i18n_taxonomy_module_implements_alter(&$implementations, $hook) {
+  // Move pathauto_i18n_taxonomy to the end of the list.
+  // By default there not available pathauto value.
+  if ($hook == 'form_alter') {
+    $group = $implementations['pathauto_i18n_taxonomy'];
+    unset($implementations['pathauto_i18n_taxonomy']);
+    $implementations['pathauto_i18n_taxonomy'] = $group;
+  }
+}
+
+/**
+ * Implements hook_action_info().
+ */
+function pathauto_i18n_taxonomy_action_info() {
+  return array(
+    'pathauto_i18n_taxonomy_generate_alias' => array(
+      'type' => 'taxonomy_term',
+      'label' => t('Enable generation of aliases for all languages'),
+      'configurable' => FALSE,
+      'behavior' => array('changes_property'),
+      'triggers' => array(
+        'any',
+      ),
+    ),
+    'pathauto_i18n_taxonomy_remove_alias' => array(
+      'type' => 'taxonomy_term',
+      'label' => t('Disable generation of aliases for all languages'),
+      'configurable' => FALSE,
+      'behavior' => array('changes_property'),
+      'triggers' => array(
+        'any',
+      ),
+    ),
+  );
+}
+
+/**
+ * Sets the status of a pathauto_i18n_status to 1.
+ *
+ * @param object $term
+ *   A taxonomy term object.
+ *
+ * @param array $context
+ *   (optional) Array of additional information about what triggered the action.
+ *   Not used for this action.
+ *
+ * @ingroup actions
+ */
+function pathauto_i18n_taxonomy_generate_alias($term, $context = array()) {
+  $term->path['pathauto_i18n_status'] = 1;
+}
+
+/**
+ * Sets the status of a pathauto_i18n_status to 0.
+ *
+ * @param object $term
+ *   A term object.
+ *
+ * @param array $context
+ *   (optional) Array of additional information about what triggered the action.
+ *   Not used for this action.
+ *
+ * @ingroup actions
+ */
+function pathauto_i18n_taxonomy_remove_alias($term, $context = array()) {
+  $term->path['pathauto_i18n_status'] = 0;
+}

+ 11 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_user/pathauto_i18n_user.info

@@ -0,0 +1,11 @@
+name = Pathauto i18n user
+description = Provides the ability to add aliases for multilanguage user
+core = 7.x
+dependencies[] = pathauto_i18n
+
+; Information added by Drupal.org packaging script on 2016-03-02
+version = "7.x-1.5"
+core = "7.x"
+project = "pathauto_i18n"
+datestamp = "1456925640"
+

+ 151 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/modules/pathauto_i18n_user/pathauto_i18n_user.module

@@ -0,0 +1,151 @@
+<?php
+/**
+ * @file
+ * Provides tools for creating multilanguage aliases for users.
+ */
+
+/**
+ * Implements hook_form_BASE_FORM_ID_alter().
+ */
+function pathauto_i18n_user_form_pathauto_patterns_form_alter(&$form, &$form_state) {
+  $languages = language_list();
+  $default_pattern = $form['user']['pathauto_user_pattern'];
+  // Remove parents handlers.
+  unset($default_pattern['#parents']);
+  $form['user']['token_help']['#weight'] = 1;
+  foreach ($languages as $language) {
+    $pattern_name = 'pathauto_user_user_' . $language->language . '_pattern';
+    $form['user'][$pattern_name] = $default_pattern;
+    $form['user'][$pattern_name]['#title'] = t('Pattern for all @language user paths', array('@language' => $language->name));
+    $form['user'][$pattern_name]['#default_value'] = variable_get($pattern_name, '');
+  }
+}
+
+/**
+ * Implements hook_form_alter().
+ */
+function pathauto_i18n_user_form_alter(&$form, &$form_state, $form_id) {
+  if ($form_id == 'user_register_form' || $form_id == 'user_profile_form') {
+    pathauto_i18n_configuration_form($form, empty($form_state['user']) ? new StdClass() : $form_state['user'], 'user', 'user');
+  }
+}
+
+/**
+ * Implements hook_user_insert().
+ */
+function pathauto_i18n_user_user_insert(&$edit, $account, $category) {
+  pathauto_i18n_init_property($account, 'user', 'user');
+  pathauto_i18n_process_entity_object($account, 'user', $account->pathauto_i18n_status, 'insert');
+}
+
+/**
+ * Implements hook_user_update().
+ */
+function pathauto_i18n_user_user_update(&$edit, $account, $category) {
+  pathauto_i18n_init_property($account, 'user', 'user');
+  pathauto_i18n_process_entity_object($account, 'user', $account->pathauto_i18n_status, 'update');
+}
+
+/**
+ * Implements hook_user_delete().
+ */
+function pathauto_i18n_user_user_delete($account) {
+  pathauto_i18n_delete_settings($account->uid, 'user');
+}
+
+/**
+ * Implements hook_user_load().
+ */
+function pathauto_i18n_user_user_load($users) {
+  // Attach pathauto i18n settings to user object.
+  foreach ($users as $user) {
+    $uids[] = $user->uid;
+  }
+
+  if (!empty($uids)) {
+    $result = pathauto_i18n_load_settings($uids, 'user');
+    foreach ($result as $record) {
+      $users[$record->entity_id]->path['pathauto_i18n_status'] = $record->path_status;
+    }
+  }
+}
+
+/**
+ * Implements hook_pathauto_alias_alter().
+ */
+function pathauto_i18n_user_pathauto_alias_alter(&$alias, &$context) {
+  $operations = array('insert', 'update', 'bulkupdate');
+  // Skip insert of alias for all languages.
+  if (!empty($context['module']) && $context['module'] == 'user' && in_array($context['op'], $operations)) {
+    $entity = $context['data']['user'];
+
+    // Run bulk update.
+    $settings = pathauto_i18n_load_settings_single($entity->uid, 'user');
+    if ($context['op'] == 'bulkupdate' && !empty($settings['path_status'])) {
+      $entity->pathauto_i18n_status = $settings['path_status'];
+      pathauto_i18n_process_entity_object($entity, 'user', $entity->pathauto_i18n_status, 'update');
+    }
+
+    if (isset($entity->pathauto_i18n_status) && $entity->pathauto_i18n_status && $context['language'] == LANGUAGE_NONE) {
+      $alias = '';
+    }
+  }
+}
+
+/**
+ * Implements hook_action_info().
+ */
+function pathauto_i18n_user_action_info() {
+  return array(
+    'pathauto_i18n_user_generate_alias' => array(
+      'type' => 'user',
+      'label' => t('Enable generation of aliases for all languages'),
+      'configurable' => FALSE,
+      'behavior' => array('changes_property'),
+      'triggers' => array(
+        'any',
+      ),
+    ),
+    'pathauto_i18n_user_remove_alias' => array(
+      'type' => 'user',
+      'label' => t('Disable generation of aliases for all languages'),
+      'configurable' => FALSE,
+      'behavior' => array('changes_property'),
+      'triggers' => array(
+        'any',
+      ),
+    ),
+  );
+}
+
+/**
+ * Sets the status of a pathauto_i18n_status to 1.
+ *
+ * @param object $user
+ *   A user object.
+ *
+ * @param array $context
+ *   (optional) Array of additional information about what triggered the action.
+ *   Not used for this action.
+ *
+ * @ingroup actions
+ */
+function pathauto_i18n_user_generate_alias($user, $context = array()) {
+  $user->pathauto_i18n_status = 1;
+}
+
+/**
+ * Sets the status of a pathauto_i18n_status to 0.
+ *
+ * @param object $user
+ *   A user object.
+ *
+ * @param array $context
+ *   (optional) Array of additional information about what triggered the action.
+ *   Not used for this action.
+ *
+ * @ingroup actions
+ */
+function pathauto_i18n_user_remove_alias($user, $context = array()) {
+  $user->pathauto_i18n_status = 0;
+}

+ 16 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/pathauto_i18n.info

@@ -0,0 +1,16 @@
+name = Pathauto i18n
+description = Provides the ability to add aliases for multilanguage content
+core = 7.x
+dependencies[] = pathauto
+dependencies[] = locale
+files[] = tests/pathauto_i18n.test.inc
+files[] = tests/pathauto_i18n_node.test
+files[] = tests/pathauto_i18n_taxonomy.test
+files[] = tests/pathauto_i18n_user.test
+
+; Information added by Drupal.org packaging script on 2016-03-02
+version = "7.x-1.5"
+core = "7.x"
+project = "pathauto_i18n"
+datestamp = "1456925640"
+

+ 46 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/pathauto_i18n.install

@@ -0,0 +1,46 @@
+<?php
+
+/**
+ * @file
+ * Install, update and uninstall functions for the Pathauto i18n module.
+ */
+
+/**
+ * Implements hook_schema().
+ */
+function pathauto_i18n_schema() {
+  $schema['pathauto_i18n'] = array(
+    'description' => 'Stores entity config for pathauto i18n.',
+    'fields' => array(
+      'entity_id' => array(
+        'description' => 'The entity id this data is attached to.',
+        'type' => 'int',
+        'unsigned' => TRUE,
+        'not null' => TRUE,
+      ),
+      'entity_type' => array(
+        'description' => 'The entity type this data is attached to.',
+        'type' => 'varchar',
+        'length' => 128,
+        'not null' => TRUE,
+        'default' => '',
+      ),
+      'bundle' => array(
+        'description' => 'Bundle to which this row belongs.',
+        'type' => 'varchar',
+        'length' => 128,
+        'not null' => TRUE,
+        'default' => '',
+      ),
+      'path_status' => array(
+        'description' => 'Boolean indicating turned on pathauto i18n.',
+        'type' => 'int',
+        'not null' => FALSE,
+        'default' => 0,
+        'size' => 'tiny',
+      ),
+    ),
+    'primary key' => array('entity_id', 'entity_type'),
+  );
+  return $schema;
+}

+ 251 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/pathauto_i18n.module

@@ -0,0 +1,251 @@
+<?php
+/**
+ * @file
+ * Provides common functions and callbacks for pathauto_i18n submodules.
+ *
+ * Does not contain any hooks or provide functionality on its own.
+ */
+
+/**
+ * Insert entity to pathauto_i18n table and generate alias if necessary.
+ */
+function pathauto_i18n_process_entity_object($entity, $entity_type, $pathauto_i18n_status, $op) {
+  // Get entity properties.
+  list($entity_id, , $bundle) = entity_extract_ids($entity_type, $entity);
+  // Insert pathauto_i18n settings.
+  if (!empty($entity_id) && !empty($bundle) && !empty($entity_type)) {
+    pathauto_i18n_insert_settings($entity_id, $entity_type, $bundle, $pathauto_i18n_status);
+    // Build aliases in it necessary.
+    if ($pathauto_i18n_status) {
+      pathauto_i18n_update_alias($entity, $entity_type, $bundle, $op);
+    }
+  }
+}
+
+/**
+ * Update the URL aliases for all available languages.
+ *
+ * @see pathauto_field_attach_form()
+ */
+function pathauto_i18n_update_alias($entity, $entity_type, $bundle, $op) {
+  // Generate aliases.
+  module_load_include('inc', 'pathauto');
+  $languages = language_list();
+  $langcode = pathauto_entity_language($entity_type, $entity);
+  $uri = entity_uri($entity_type, $entity);
+  foreach ($languages as $language) {
+    // Skipping a creating alias for node language, it created automatically.
+    if (empty($entity->language) || $langcode != $language->language) {
+      pathauto_create_alias(
+        $entity_type,
+        $op,
+        $uri['path'],
+        array($entity_type => $entity),
+        $bundle,
+        $language->language
+      );
+    }
+  }
+}
+
+/**
+ * Returns the configured default value for a bundle.
+ */
+function pathauto_i18n_get_bundle_default($entity_type, $bundle, $default = 1) {
+  return variable_get('pathauto_i18n_default_' . $entity_type . '_' . $bundle, $default);
+}
+
+/**
+ * Sets a default value for a bundle.
+ */
+function pathauto_i18n_set_bundle_default($entity_type, $bundle, $value) {
+  variable_set('pathauto_i18n_default_' . $entity_type . '_' . $bundle, $value);
+}
+
+/**
+ * Attach Pathauto i18n field configuration to form.
+ *
+ * @param array $form
+ *   Form to attach.
+ *
+ * @param object $entity
+ *   Entity object.
+ *
+ * @param string $entity_type
+ *   Entity type.
+ *
+ * @see pathauto_field_attach_form()
+ */
+function pathauto_i18n_configuration_form(&$form, $entity, $entity_type, $bundle) {
+  $access = user_access('create url aliases') || user_access('administer url aliases');
+  if ($entity && $access) {
+    $default = pathauto_i18n_get_bundle_default($entity_type, $bundle);
+    $form['path']['pathauto_i18n_status'] = array(
+      '#type' => 'checkbox',
+      '#title' => t('Generate automatic URL alias for all languages'),
+      '#description' => t('Allows you to generate aliases for all available languages.', array('@entity_type' => $entity_type)),
+      '#default_value' => isset($entity->path['pathauto_i18n_status']) ? $entity->path['pathauto_i18n_status'] : $default,
+      '#weight' => -0.99,
+    );
+    $form['path']['pathauto_i18n_undefined_language_tip'] = array(
+      '#type' => 'item',
+      '#markup' => t('URL alias for "Language neutral" <strong>won\'t be created</strong>, because you use automatic alias.') . '</strong>',
+      '#weight' => -0.98,
+      '#states' => array(
+        'visible' => array(
+          'select[name="language"]' => array('value' => LANGUAGE_NONE),
+          'input[name="path[pathauto]"]' => array('checked' => TRUE),
+          'input[name="path[pathauto_i18n_status]"]' => array('checked' => TRUE),
+        ),
+      ),
+    );
+
+    $form['path']['pathauto_i18n_undefined_language_custom_tip'] = array(
+      '#type' => 'item',
+      '#markup' => t('URL alias for "Language neutral" <strong>will be created</strong>, because you use custom alias.'),
+      '#weight' => -0.98,
+      '#states' => array(
+        'visible' => array(
+          'select[name="language"]' => array('value' => LANGUAGE_NONE),
+          'input[name="path[pathauto]"]' => array('checked' => FALSE),
+          'input[name="path[pathauto_i18n_status]"]' => array('checked' => TRUE),
+        ),
+      ),
+    );
+  }
+}
+
+/**
+ * Insert settings for entity.
+ */
+function pathauto_i18n_insert_settings($entity_id, $entity_type, $bundle, $path_status) {
+  db_merge('pathauto_i18n')
+    ->key(array('entity_id' => $entity_id, 'entity_type' => $entity_type))
+    ->fields(
+      array(
+        'entity_id' => $entity_id,
+        'entity_type' => $entity_type,
+        'bundle' => $bundle,
+        'path_status' => $path_status,
+      )
+    )
+    ->execute();
+}
+
+/**
+ * Load settings for entity.
+ */
+function pathauto_i18n_load_settings($ids, $entity_type) {
+  return db_select('pathauto_i18n', 'p')
+    ->fields('p', array(
+      'entity_id',
+      'path_status',
+    ))
+    ->condition('p.entity_id', $ids, 'IN')
+    ->condition('p.entity_type', $entity_type)
+    ->execute();
+}
+
+/**
+ * Load settings for entity.
+ */
+function pathauto_i18n_load_settings_single($id, $entity_type) {
+  return db_select('pathauto_i18n', 'p')
+    ->fields('p', array(
+        'entity_id',
+        'path_status',
+      ))
+    ->condition('p.entity_id', $id)
+    ->condition('p.entity_type', $entity_type)
+    ->execute()
+    ->fetchAssoc();
+}
+
+/**
+ * Delete settings for certain entity.
+ */
+function pathauto_i18n_delete_settings($entity_id, $entity_type) {
+  db_delete('pathauto_i18n')
+    ->condition('entity_id', $entity_id)
+    ->condition('entity_type', $entity_type)
+    ->execute();
+}
+
+
+/**
+ * Implements hook_form_FORM_ID_alter().
+ *
+ * Make pathauto_ignore_words global and add fields for languages.
+ */
+function pathauto_i18n_form_pathauto_settings_form_alter(&$form, &$form_state, $form_id) {
+  module_load_include('inc', 'pathauto');
+  $languages = language_list();
+  $form['pathauto_ignore_words']['#title'] = t('Strings to Remove - @language', array('@language' => t('Global')));
+
+  foreach ($languages as $language) {
+    $key = 'pathauto_ignore_words_' . $language->language . '_language';
+    $form[$key] = array(
+      '#type' => 'textarea',
+      '#title' => t('Strings to Remove - @language', array('@language' => $language->name)),
+      '#default_value' => variable_get($key, ''),
+      '#wysiwyg' => FALSE,
+    );
+  }
+}
+
+/**
+ * Clean up a string segment for certain language.
+ */
+function pathauto_i18n_cleanstring($language, $string) {
+  $ignore_words = variable_get('pathauto_ignore_words_' . $language . '_language', '');
+  $ignore_words_regex = preg_replace(array('/^[,\s]+|[,\s]+$/', '/[,\s]+/'), array('', '\b|\b'), $ignore_words);
+  if ($ignore_words_regex) {
+    $ignore_words_regex = '\b' . $ignore_words_regex . '\b';
+    if (function_exists('mb_eregi_replace')) {
+      $ignore_words_callback = 'mb_eregi_replace';
+    }
+    else {
+      $ignore_words_callback = 'preg_replace';
+      $ignore_words_regex = '/' . $ignore_words_regex . '/i';
+    }
+
+    if (!empty($ignore_words_regex) && !empty($ignore_words_callback)) {
+      $words_removed = $ignore_words_callback($ignore_words_regex, '', $string);
+      if (drupal_strlen(trim($words_removed)) > 0) {
+        $string = $words_removed;
+      }
+    }
+  }
+
+  return $string;
+}
+
+/**
+ * Implements hook_tokens_alter().
+ */
+function pathauto_i18n_tokens_alter(&$replacements, array $context) {
+  // @todo need tests.
+  if (!empty($context['options']['pathauto'])) {
+    foreach ($context['tokens'] as $name => $original) {
+      if (!empty($replacements[$original]) && !empty($context['options']['language']->language) && $context['options']['language']->language != LANGUAGE_NONE) {
+        $replacements[$original] = pathauto_i18n_cleanstring($context['options']['language']->language, $replacements[$original]);
+      }
+    }
+  }
+}
+
+/**
+ * Init the property.
+ */
+function pathauto_i18n_init_property(&$entity, $type, $bundle) {
+  $default = pathauto_i18n_get_bundle_default($type, $bundle);
+  switch ($type) {
+    case 'user':
+      $entity->pathauto_i18n_status = isset($entity->pathauto_i18n_status) ? $entity->pathauto_i18n_status : $default;
+      break;
+
+    default:
+      $entity->path['pathauto_i18n_status'] = isset($entity->path['pathauto_i18n_status']) ? $entity->path['pathauto_i18n_status'] : $default;
+      break;
+  }
+}

+ 97 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n.test.inc

@@ -0,0 +1,97 @@
+<?php
+/**
+ * @file
+ * Tests for the pathauto_i18n node module.
+ */
+
+/**
+ * Test functionality for nodes when language selected.
+ */
+class Pathautoi18nTest extends DrupalWebTestCase {
+  /**
+   * Fully loaded user object of an admin user that has required access rights.
+   *
+   * @var object
+   */
+  protected $admin;
+
+  /**
+   * Default language.
+   */
+  protected $contentLanguage;
+
+  /**
+   * Available languages.
+   */
+  protected $availableLanguages;
+
+  /**
+   * Title.
+   */
+  protected $title;
+
+  /**
+   * Prepare test.
+   */
+  public function prepareTest($modules) {
+    $modules[] = 'pathauto_i18n';
+    parent::setUp($modules);
+
+    $this->admin = $this->drupalCreateUser(array(
+      'access administration pages',
+      'administer nodes',
+      'administer languages',
+      'administer content types',
+      'administer url aliases',
+      'create url aliases',
+      'administer pathauto',
+    ));
+
+    $this->drupalLogin($this->admin);
+
+    // Set content language and all available languages.
+    $this->contentLanguage = 'en';
+    $this->availableLanguages = array(
+      'en',
+      'fr',
+      'de',
+    );
+
+    $this->title = 'pathautoi18n';
+    foreach ($this->availableLanguages as $language) {
+      if ($language != $this->contentLanguage) {
+        $this->drupalPost('admin/config/regional/language/add', array('langcode' => $language), t('Add language'));
+      }
+    }
+    drupal_static_reset('language_list');
+
+    // Enabel multilingual support for content type.
+    $this->drupalPost('admin/structure/types/manage/article', array('language_content_type' => 1), t('Save content type'));
+  }
+
+  /**
+   * Set settings to test cleanstring.
+   */
+  public function setCleanStringSettings() {
+    $data = array();
+    foreach ($this->availableLanguages as $language) {
+      $data['pathauto_ignore_words_' . $language . '_language'] = $language;
+    }
+
+    $this->drupalPost('admin/config/search/path/settings', $data, t('Save configuration'));
+  }
+
+  /**
+   * Return suffix for certain language.
+   */
+  public function getCleanStringSuffix($skip_language) {
+    $suffix = array();
+    foreach ($this->availableLanguages as $language) {
+      if ($language != $skip_language) {
+        $suffix[] = $language;
+      }
+    }
+
+    return implode('-', $suffix);
+  }
+}

+ 151 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n_node.test

@@ -0,0 +1,151 @@
+<?php
+/**
+ * @file
+ * Tests for the pathauto_i18n node module.
+ */
+
+module_load_include('inc', 'pathauto_i18n', 'tests/pathauto_i18n.test');
+/**
+ * Test functionality for nodes.
+ */
+class Pathautoi18nNodeTest extends Pathautoi18nTest {
+
+  /**
+   * GetInfo method.
+   */
+  public static function getInfo() {
+    return array(
+      'name' => 'Pathauto i18n node',
+      'description' => 'Ensure that the Pathauto i18n node works.',
+      'group' => 'Pathauto i18n',
+    );
+  }
+
+  /**
+   * SetUp method.
+   */
+  public function setUp() {
+    $modules[] = 'pathauto_i18n_node';
+    $this->prepareTest($modules);
+
+    // Configure patterns for all language for easy testing.
+    $edit = array(
+      'pathauto_node_article_pattern' => 'all/[node:title]',
+      'pathauto_node_article_und_pattern' => 'neutral/[node:title]',
+    );
+
+    foreach ($this->availableLanguages as $language) {
+      $edit['pathauto_node_article_' . $language . '_pattern'] = $language . '/[node:title]';
+    }
+    $this->drupalPost('admin/config/search/path/patterns', $edit, t('Save configuration'));
+  }
+
+  /**
+   * Test nodes with automatic alias for certain language.
+   */
+  public function testAutomaticAliasLanguageSelected() {
+    $this->createNode($this->contentLanguage, TRUE, TRUE);
+    // Check aliases.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $alias = $language . '/' . $this->title;
+      $this->assertText($alias, 0, "Exist alias '$alias' for language '$language'.");
+    }
+  }
+
+  /**
+   * Test nodes with custom alias for certain language.
+   */
+  public function testCustomAliasLanguageSelected() {
+    $custom_alias = 'custom/' . $this->title;
+    $this->createNode($this->contentLanguage, TRUE, FALSE, $custom_alias);
+    // Check aliases and custom alias.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      if ($language == 'en') {
+        $alias = $language . '/' . $this->title;
+        $this->assertText($custom_alias, 0, "Exist custom alias '$custom_alias'.");
+        $this->assertNoText($alias, 0, "Alias '$alias' for language '$language' not exist.");
+      }
+      else {
+        $alias = $language . '/' . $this->title;
+        $this->assertText($alias, 0, "Exist alias '$alias' for language '$language'.");
+      }
+    }
+  }
+
+  /**
+   * Test nodes with automatic alias for undefined language.
+   */
+  public function testAutomaticAliasUndefinedLanguage() {
+    $this->contentLanguage = LANGUAGE_NONE;
+    $this->testAutomaticAliasLanguageSelected();
+    $neutral_alias = 'neutral/' . $this->title;
+    $this->assertNoText($neutral_alias, 0, "Alias '$neutral_alias' for undefined language not exist.");
+  }
+
+  /**
+   * Test nodes with custom alias for undefined language.
+   */
+  public function testCustomAliasUndefinedLanguage() {
+    $custom_alias = 'custom/' . $this->title;
+    $neutral_alias = 'neutral/' . $this->title;
+    $this->createNode(LANGUAGE_NONE, TRUE, FALSE, $custom_alias);
+
+    // Check aliases and custom alias.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $alias = $language . '/' . $this->title;
+      $this->assertText($alias, 0, "Exist alias '$alias' for language '$language'.");
+    }
+
+    $this->assertText($custom_alias, 0, "Exist custom alias '$custom_alias'.");
+    $this->assertNoText($neutral_alias, 0, "Alias '$neutral_alias' for undefined language not exist.");
+  }
+
+  /**
+   * Test clearing of string.
+   */
+  public function testCleanString() {
+    // Set appropriate title which will allow us remove parts of path.
+    $initial_title = $this->title;
+    $this->title .= ' ' . implode(' ', $this->availableLanguages);
+
+    $this->setCleanStringSettings();
+    $this->createNode($this->contentLanguage, TRUE, TRUE);
+
+    // Check aliases.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $suffix = $this->getCleanStringSuffix($language);
+      $alias = $language . '/' . $initial_title . '/' . $suffix;
+      $this->assertNoText($alias, 0, "Exist alias '$alias' for language '$language' with excluded string '$language'.");
+    }
+  }
+
+  /**
+   * Helper to create node.
+   */
+  public function createNode($language, $pathauto_i18n_status, $pathauto, $alias = FALSE) {
+    $settings = array(
+      'type' => 'article',
+      'title' => $this->title,
+      'body' => array(
+        $language => array(
+          array(
+            $this->randomName(64),
+          ),
+        ),
+      ),
+      'language' => $language,
+      'path' => array(
+        'pathauto_i18n_status' => $pathauto_i18n_status,
+        'pathauto' => $pathauto,
+      ),
+    );
+    if (!empty($alias)) {
+      $settings['path']['alias'] = $alias;
+    }
+    $node = $this->drupalCreateNode($settings);
+  }
+}

+ 116 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n_taxonomy.test

@@ -0,0 +1,116 @@
+<?php
+/**
+ * @file
+ * Tests for the pathauto_i18n taxonomy module.
+ */
+
+module_load_include('inc', 'pathauto_i18n', 'tests/pathauto_i18n.test');
+define('PATHAUTO_I18N_TAG_VOCABULARU_MACHINE_NAME', 'tags');
+
+/**
+ * Test functionality for taxonomy.
+ */
+class Pathautoi18nTaxonomyTest extends Pathautoi18nTest {
+
+  /**
+   * GetInfo method.
+   */
+  public static function getInfo() {
+    return array(
+      'name' => 'Pathauto i18n taxonomy',
+      'description' => 'Ensure that the Pathauto i18n taxonomy works.',
+      'group' => 'Pathauto i18n',
+    );
+  }
+
+  /**
+   * SetUp method.
+   */
+  public function setUp() {
+    $modules[] = 'pathauto_i18n_taxonomy';
+    $this->prepareTest($modules);
+
+    // Configure patterns for all language for easy testing.
+    $edit = array(
+      'pathauto_taxonomy_term_pattern' => 'all/[term:name]',
+      'pathauto_taxonomy_term_tags_pattern' => 'neutral/[term:name]',
+    );
+
+    foreach ($this->availableLanguages as $language) {
+      $edit['pathauto_taxonomy_term_tags_' . $language . '_pattern'] = $language . '/[term:name]';
+    }
+    $this->drupalPost('admin/config/search/path/patterns', $edit, t('Save configuration'));
+  }
+
+  /**
+   * Test taxonomy terms with automatic alias.
+   */
+  public function testAutomaticAliasTaxonomy() {
+    $this->createTaxonomyTerm(TRUE, TRUE);
+    // Check aliases.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $alias = $language . '/' . $this->title;
+      $this->assertText($alias, 0, "Exist alias '$alias' for language '$language'.");
+    }
+  }
+
+  /**
+   * Test taxonomy terms with custom alias for certain language.
+   */
+  public function testCustomAliasTaxonomy() {
+    $custom_alias = 'custom/' . $this->title;
+    $neutral_alias = 'neutral/' . $this->title;
+    $this->createTaxonomyTerm(TRUE, TRUE, $custom_alias);
+    // Check aliases and custom alias.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $alias = $language . '/' . $this->title;
+      $this->assertText($alias, 0, "Exist alias '$alias' for language '$language'.");
+    }
+
+    $this->assertText($custom_alias, 0, "Exist custom alias '$custom_alias'.");
+    $this->assertNoText($neutral_alias, 0, "Alias '$neutral_alias' for undefined language not exist.");
+  }
+
+  /**
+   * Test clearing of string.
+   */
+  public function testCleanString() {
+    // Set appropriate title which will allow us remove parts of path.
+    $initial_title = $this->title;
+    $this->title .= ' ' . implode(' ', $this->availableLanguages);
+
+    $this->setCleanStringSettings();
+    $this->createTaxonomyTerm(TRUE, TRUE);
+
+    // Check aliases.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $suffix = $this->getCleanStringSuffix($language);
+      $alias = $language . '/' . $initial_title . '/' . $suffix;
+      $this->assertNoText($alias, 0, "Exist alias '$alias' for language '$language' with excluded string '$language'.");
+    }
+  }
+
+  /**
+   * Helper to create taxonomy term.
+   */
+  public function createTaxonomyTerm($pathauto_i18n_status, $pathauto, $alias = FALSE) {
+    $vocabulary = taxonomy_vocabulary_machine_name_load(PATHAUTO_I18N_TAG_VOCABULARU_MACHINE_NAME);
+    $term = new stdClass();
+    $term->name = $this->title;
+    $term->description = $this->title;
+    // Use the first available text format.
+    $term->format = db_query_range('SELECT format FROM {filter_format}', 0, 1)->fetchField();
+    $term->vid = $vocabulary->vid;
+    $term->path = array(
+      'pathauto_i18n_status' => $pathauto_i18n_status,
+      'pathauto' => $pathauto,
+    );
+    if (!empty($alias)) {
+      $term->path['alias'] = $alias;
+    }
+    taxonomy_term_save($term);
+  }
+}

+ 89 - 0
sites/all/modules/contrib/localisation/pathauto_i18n/tests/pathauto_i18n_user.test

@@ -0,0 +1,89 @@
+<?php
+/**
+ * @file
+ * Tests for the pathauto_i18n user module.
+ */
+
+module_load_include('inc', 'pathauto_i18n', 'tests/pathauto_i18n.test');
+
+/**
+ * Test functionality for user.
+ */
+class Pathautoi18nUserTest extends Pathautoi18nTest {
+
+  /**
+   * GetInfo method.
+   */
+  public static function getInfo() {
+    return array(
+      'name' => 'Pathauto i18n user',
+      'description' => 'Ensure that the Pathauto i18n user works.',
+      'group' => 'Pathauto i18n',
+    );
+  }
+
+  /**
+   * SetUp method.
+   */
+  public function setUp() {
+    $modules[] = 'pathauto_i18n_user';
+    $this->prepareTest($modules);
+
+    // Configure patterns for all language for easy testing.
+    $edit = array(
+      'pathauto_user_pattern' => 'neutral/users/[user:name]',
+    );
+
+    foreach ($this->availableLanguages as $language) {
+      $edit['pathauto_user_user_' . $language . '_pattern'] = $language . '/users/[user:name]';
+    }
+    $this->drupalPost('admin/config/search/path/patterns', $edit, t('Save configuration'));
+  }
+
+  /**
+   * Test user alias.
+   */
+  public function testUserAlias() {
+    drupal_static_reset('pathauto_pattern_load_by_entity');
+    $this->createUser();
+    // Check aliases.
+    $this->drupalGet('admin/config/search/path/list/users');
+    foreach ($this->availableLanguages as $language) {
+      $alias = $language . '/users/' . $this->title;
+      $this->assertText($alias, 0, "Exist alias '$alias' for language '$language'.");
+    }
+  }
+
+  /**
+   * Test clearing of string.
+   */
+  public function testCleanString() {
+    // Set appropriate title which will allow us remove parts of path.
+    $initial_title = $this->title;
+    $this->title .= ' ' . implode(' ', $this->availableLanguages);
+
+    $this->setCleanStringSettings();
+    $this->createUser();
+
+    // Check aliases.
+    $this->drupalGet('admin/config/search/path');
+    foreach ($this->availableLanguages as $language) {
+      $suffix = $this->getCleanStringSuffix($language);
+      $alias = $language . '/' . $initial_title . '/' . $suffix;
+      $this->assertNoText($alias, 0, "Exist alias '$alias' for language '$language' with excluded string '$language'.");
+    }
+  }
+
+  /**
+   * Helper to create users.
+   */
+  public function createUser() {
+    $edit = array();
+    $edit['name'] = $this->title;
+    $edit['mail'] = $edit['name'] . '@sanchiz.net';
+    $edit['pass'] = user_password();
+    $edit['status'] = 1;
+    $edit['pathauto_i18n_status'] = 1;
+    user_save(drupal_anonymous_user(), $edit);
+  }
+}