This topic explains how to handle strings in Python and provides examples of operations that deal with strings. It contains the following sections:
 Accessing an Individual Character of a String
Accessing an Individual Character of a String
 Searching for Characters and Substrings
Searching for Characters and Substrings
 Removing Extra Spaces From a String
Removing Extra Spaces From a String
Basics
A string is a sequence of symbols or digits. Strings are among the most frequently used data types. Like any other data types, strings in TestComplete are represented as OLE-compatible variants.
In Python, a sequence of literal characters enclosed in a matching pair of quotation marks is recognized as a string. Quotation marks can be single ('), double (") or triple quotes (""" or '''). The quotation characters are allowed within a string if they do not match the enclosing pair. All the strings below are valid:
Python
str2 = 'It came out of night fog and looked weird: no sails, no lights and no crew.';
str3 = "'Ahoy! Is there anyone?' - the captain cried.";
str4 = """"Nobody." - was the answer."""
To deal with strings, TestComplete has a special aqString scripting object. The object is available for all supported scripting languages, so that you can use it to operate with string values regardless of the chosen language. 
| Method, Property | Description | 
|---|---|
| aqString.AddListItem | Adds a new item to a string list. | 
| aqString.ChangeListItem | Changes the value of the string list item with the given index. | 
| aqString.Compare | Compares two specified strings. | 
| aqString.Concat | Concatenates two specified strings. | 
| aqString.DeleteListItem | Removes an item with the given index from a string list. | 
| aqString.Find | Searches for a substring within the given string. Use this method instead of the obsolete aqString.Contains. | 
| aqString.Format | Generates a formatted string. | 
| aqString.GetChar | Retrieves a single character from the input string. | 
| aqString.GetLength | Returns the number of characters in a string. | 
| aqString.GetListItem | Returns an individual item from the list passed through the input string. | 
| aqString.GetListLength | Returns the number of items in the string list. | 
| aqString.Insert | Inserts one string to another at the specified position. | 
| aqString.ListSeparator | Specifies a character used to separate individual values in a list. | 
| aqString.Quote | Encloses the specified string in quotes. | 
| aqString.QuoteSymbol | Specifies a symbol used as a quotation mark. | 
| aqString.Remove | Removes a number of characters from the input string. | 
| aqString.Replace | Replaces all the occurrences of one substring with another substring. | 
| aqString.SubString | Retrieves a substring from the input string. | 
| aqString.ToLower | Converts the specified string to lower case. | 
| aqString.ToUpper | Converts the specified string to upper case. | 
| aqString.Trim | Removes spaces and control characters from the specified string. | 
| aqString.Unquote | Converts a quoted string to an unquoted string. | 
Another scripting object that is useful for string manipulation is aqConvert. This object has several methods that convert values of different types to a string representation and vice versa.
| Method | Description | 
|---|---|
| aqConvert.CurrencyToFormatStr | Converts a currency value to a string using the specified format settings. | 
| aqConvert.CurrencyToStr | Converts a currency value to a string. | 
| aqConvert.DateTimeToFormatStr | Converts the given date value to a string using the specified format. | 
| aqConvert.DateTimeToStr | Converts the given date value to a string. | 
| aqConvert.FloatToStr | Converts a floating-point value to a string. | 
| aqConvert.IntToStr | Converts the given number into a string. | 
| aqConvert.StrToCurrency | Converts the specified string to a currency value. | 
| aqConvert.StrToDate | Converts the specified string to a date value. | 
| aqConvert.StrToDateTime | Converts the specified string to a date/time value. | 
| aqConvert.StrToFloat | Converts the specified string to a floating-point value. | 
| aqConvert.StrToInt | Converts the specified string to an integer value. | 
| aqConvert.StrToInt64 | Converts the specified string to a long integer value. | 
| aqConvert.StrToTime | Converts the specified string to a time value. | 
| aqConvert.VarToStr | Converts the specified variant value to a string. | 
Furthermore, Python has its own inherent object, string, which is a wrapper for the string data type. Strings that are used in TestComplete are compatible with this object. Moreover, string objects are created implicitly for every string. That is, you can call object methods not only for object instances but also for any TestComplete string. For example, the following method call is correct:
A detailed description of the string object is provided in Python Wiki. The table below lists the object's major properties and methods:
| Property, Method | Description | 
|---|---|
| capitalize() | Method. Capitalizes the first letter in a string. | 
| center(width, fillchar) | Method. Returns a center-aligned string. Spaces are used for formatting. | 
| count(str, beg= 0,end=len(string)) | Method. Counts how many times stroccurs in a string or substring if the starting index (beg) and the ending index (end) are specified. | 
| decode(encoding='UTF-8',errors='strict') | Method. Decodes a string by using the specified codec. Uses the default string encoding if no encoding is specified. | 
| encode(encoding='UTF-8',errors='strict') | Method. Returns an encoded string. If an error occurs, the method displays the ValueErrormessage unlessthe errorsparameter is specified as 'ignore' or 'replace'. | 
| endswith(suffix, beg=0, end=len(string)) | Method. Determines if a string or a substring (if the starting index ( beg) and the ending index (end) are specified) ends with a suffix. Returns True on success and False otherwise. | 
| expandtabs(tabsize=8) | Method. Replaces tabs in a string with multiple spaces. Uses 8 spaces per tab if tabsizeis not specified. | 
| find(str, beg=0 end=len(string)) | Method. Determines if stroccurs in a string or substring if the starting index (beg) and the ending index (end) are specified. Returns the index of the first symbol in the match ifstris found, and -1 otherwise. | 
| index(str, beg=0, end=len(string)) | Method. The same as find(), but raises an exception ifstris not found. | 
| isalnum() | Property. Returns True if a string has at least 1 character and all the characters are alphanumeric. False otherwise. | 
| isalpha() | Property. Returns True if a string has at least 1 character and all the characters are alphabetic. False otherwise. | 
| isdigit() | Property. Returns True if a string contains only digits, and False otherwise. | 
| islower() | Property. Returns True if all the characters in a string are lowercase. | 
| isspace() | Property. Returns True if a string contains only whitespace characters, and False otherwise. | 
| istitle() | Property. Returns True if title case is used in a string (the first letter of each word is in uppercase), and False otherwise. | 
| isupper() | Property. Returns True if all the characters in a string are in uppercase. | 
| join(seq) | Method. Combines the elements in the seqsequence. Inserts a separator between the elements. | 
| len(string) | Property. Returns the length of the string. | 
| ljust(width[, fillchar]) | Method. Returns a left-aligned string. Spaces are used for formatting. | 
| lower() | Method. Converts all the letters in a string to lowercase letters. | 
| lstrip() | Method. Removes all leading whitespaces from a string. | 
| maketrans() | Method. Returns a translation table used to translate strings. | 
| max(str) | Property. Returns the lexicographically highest character from the string. For example, max("smartbear")will return t. | 
| min(str) | Property. Returns the lexicographically lowest character from the string. For example, min("smartbear")will return a. | 
| replace(old, new [, max]) | Method. Replaces the occurrences of an old substring in a string with a new substring. The number of replacements is specified by the maxargument. | 
| rfind(str, beg=0,end=len(string)) | Method. The same as find(), but searches backwards in a string. | 
| rindex( str, beg=0, end=len(string)) | Method. The same as index(), but searches backwards in a string. | 
| rjust(width,[, fillchar]) | Method. Returns a right-aligned string. Spaces are used for formatting. | 
| split(str="", num=string.count(str)) | Method. Splits a string by using the strdelimiter (ifstris not specified, spaces are used) and returns a list of substrings. Thenumargument specifies the number of substrings. | 
| splitlines( num=string.count('\n')) | Method. Splits a string by new line characters and returns a list of lines. New line characters are removed from these lines. The numargument specifies the number of lines. | 
| startswith(str, beg=0,end=len(string)) | Method. Determines if a string or a substring (if the starting index begand the ending indexendare specified) starts with the substringstr. Returns True if it does, and False otherwise. | 
| strip([chars]) | Method. Removes all leading and trailing whitespaces from a string. | 
| swapcase() | Method. Inverts the case of all the letters in a string. | 
| title() | Property. Returns the titlecased version of the string (all the words begin with uppercase letters and the other letters are in lowercase). | 
| translate(table, deletechars="") | Method. Translates the words specified in the translation table. | 
| upper() | Method. Converts lowercase letters in a string to uppercase letters. | 
| zfill (width) | Property. Returns a string with zeros added at the beginning of the string to reach the specified width. | 
| isdecimal() | Property. Returns True if the string contains only decimal characters, and False otherwise. | 
Special Characters
There are several pairs of symbols used to represent special characters in Python strings. All of them start with a backslash character (\), and are often called escape characters. The following escape characters are allowed in Python.
| Character sequence | Description | 
|---|---|
| \newline | Ignored new line | 
| \' | Single quote | 
| \" | Single quote | 
| \a | Bell | 
| \b | Backspace | 
| \f | Formfeed | 
| \n | Linefeed | 
| \r | Carriage Return | 
| \t | Horizontal Tab | 
| \v | Vertical Tab | 
| \ooo | ASCII character with octal value ooo | 
| \xhh... | ASCII character with hex value hh... | 
String Prefixes
In Python, you can prefix a string with a special character to specify the way in which it will be processed. In TestComplete, you can use only one prefix for a string. If you need to process a string in multiple ways, process it several times, or use Python methods to convert it.
| Prefix | Description | 
| b | A string will be stored by using an array of bytes. | 
| r | All escape codes in a string will be ignored. All characters will be escaped. | 
| u | A string will be processed as a Unicode string. | 
Getting the String Length
To obtain the total number of characters in a string, you can either call the len Python method, or the aqString.GetLength method. The character position in Python strings is zero-based, so the maximum position number in a string is Length-1. The code below demonstrates both ways of obtaining the string length:
Python
def StringLengthDemo():
  aString = "Some text"
  Log.Message("The string is " + str(len(aString)) + " character(s) long.")
  Log.Message("The string is " + str(aqString.GetLength(aString)) + " character(s) long.")|  | Using TestComplete, you can limit the length of string parameters returned by functions of the tested application. For this purpose, use the Maximum string length of [out] parameters project property. If the length of the string returned from the application via one of its out parameters exceeds the property’s value, TestComplete treats the returned string as a null one. | 
Concatenating Strings
Several strings can be concatenated to form another string. The aqString object has a special (aqString.Concat) method that performs this operation.
However, it is more convenient to use the addition operator (+) for this purpose. To append an numeric value, convert it to a string by using the str method.
The sample code below demonstrates how to use all three techniques:
Comparing Strings
You may often need to compare two strings during testing. Generally, a test procedure obtains text data (user input, file contents, property value and so on) and then compares it with the expected data. In Python, the comparison operator (==) can check if the strings are identical. For the TestComplete aqString object, there is the aqString.Compare method.
The aqString.Compare method has three parameters: two of them, String 1 and String2, specify the strings to be compared, while the last parameter defines whether the comparison should be case-sensitive or not.
Both approaches can be used to compare strings containing national symbols, for example, characters with diacritical marks ( ,
 ,  ,
 ,  and others).
  and others).
The code below demonstrates how to use both approaches:
Python
def StringComparison():
  # Using the method of the native String object
  if "Abra" == "abra":
    Log.Message("Abra is equal to abra")
  else:
    Log.Message("Abra is not equal to abra")
  if "abra" == "Abra":
    Log.Message("abra is equal to Abra")
  else:
    Log.Message("abra is not equal to Abra")
  if "abra" == "abra":
    Log.Message("abra is equal to abra")
  else:
    Log.Message("abra is not equal to abra")
  # Using the method of the aqString object
  Log.Message("aqString.Compare(\"Abra\", \"abra\", False): " + str(aqString.Compare("Abra","abra", False)));
  Log.Message("aqString.Compare(\"Abra\", \"abra\", True): " + str(aqString.Compare("Abra","abra", True)));
  # Dealing with native-language characters
  # The following code assumes that French layout is installed on computer
  
  # French layout
  LangID = 0x040C
  LocaleID = "0000040c"
    
  if aqEnvironment.IsLanguageSupported(LangID) and aqEnvironment.SetKeyboardLayout(Sys.Process("TestComplete").Id, LocaleID):
    
    if "français" == "Français":
        Log.Message("français is equal to Français")
    else:
      Log.Message("français is not equal to Français")
    if "français" == "français":
      Log.Message("français is equal to français")
    else:
      Log.Message("français is not equal to français")
    if "français" == "francais":
      Log.Message("français is equal to francais")
    else:
      Log.Message("français is not equal to francais")
 
    Log.Message("aqString.Compare(\"français\", \"Français\", False): "+str(aqString.Compare("français", "Français", False)))
    Log.Message("aqString.Compare(\"français\", \"Francais\", False): "+str(aqString.Compare("français", "Francais", False)))
    Log.Message("aqString.Compare(\"français\", \"Français\", True): "+str(aqString.Compare("français", "Français", True)))Accessing an Individual Character of a String
The Python scripting language does not have a data type used for storing single symbols. However, it is not a problem, since strings can contain series of characters and individual characters. You can access any symbol in a string by specifying its valid index. Moreover, you can extract a single character from a string using the aqString.GetChar method. The sample routine below demonstrates how to extract symbols from strings. It posts text to the TestComplete log in two different ways: as a whole string and by single letters.
Python
def StringByLetter():
  aString = "Per aspera ad astra"
  Log.Message("The string is : " + aString)
  Log.Message("And now this text letter by letter using aqString.GetChar:")
  for i in range(0, len(aString)):
    Log.Message(aqString.GetChar(aString, i));
  Log.Message("And now this text letter by letter using Python syntax:")
  for char in aString:
    Log.Message(char)Searching for Characters and Substrings
One of the most common tasks that you may need to perform when working with string values is determining whether specific text is part of a string. To perform such tasks, use a conditional statement or the aqString.Find method. If the specified substring was found, the method returns the number of the first occurrence of the substring within the source string. If the specified substring was not found, the method returns -1:
Python
def StringOccurrenceDemo():
  aString = "Per aspera ad astra"
  aSubString = "astra"
  Res = aqString.Find(aString, aSubString)
  if  Res != -1:
    Log.Message("A substring '" + aSubString + "' was found in string '" + aString + "' at position " + aqConvert.IntToStr(Res));
  else:
    Log.Message("There are no occurrences of '" + aSubString + "' in '" + aString + "'.")You can also get the position of the specified substring by using Python's native find method. If the substring was found, the method returns the initial position (from 0) of the first substring match. If no occurrences were found, the method returns -1. The code below demonstrates how to do this:
Python
def TextPosDemo():
  aString = "Per aspera ad astra"
  aSubString = "astra"
  findpos = aString.find(aSubString)
  
  if findpos != -1:
    Log.Message("A substring '" + aSubString + "' was found at position " + str(findpos))
  else:
    Log.Message("There are no occurrences of '" + aSubString + "' in '" + aString + "'.")Getting a Substring
Python strings can be accessed as arrays. This means that to get a substring, you can just use the array syntax ([x:y]).
TestComplete provides the aqString object that has the SubString method. This method returns the text fragment between the starting and ending character positions. The character at the ending position is not included. The position is zero-based. If the starting position is greater than the ending position, then these values are swapped. Negative values are considered as zeros.
The sample code below demonstrates how to use these approaches:
Python
def GetStringDemo():
  Str = "0123456789";
  Log.Message("The Python array syntax demo:")
  Log.Message(Str[2:7]) # Posts "23456"
  Log.Message(Str[7:2]) # Posts "23456"
  Log.Message(Str[-2:7]) # Posts "0123456"
  Log.Message(Str[2:2]) # Posts ""
  Log.Message("The 'aqString.SubString' method demo:")
  Log.Message(aqString.SubString(Str, 2, 5)) # Posts "23456"
  Log.Message(aqString.SubString(Str, 2, 0)) # Posts ""Splitting Strings
Sometimes, you may need to make several strings out of a single string. This operation splits a string into substrings. This can be done by using the Python split method.  This method searches the string for delimiter characters (defined either as a string or a regular expression), splits the string and returns a list containing the resulting strings. Also, you can limit the list length using the third parameter. It is optional and can be omitted. The same method can be used to split a string into substrings, sentences and even separate words. This depends on the specified delimiter. The first sample routine below uses a space character as a delimiter to extract words from a string and the second routine splits a string by line breaks:
Python
def SplitDemo1():
    s = "Better late than never but better never late."
    # Split at each space character.
    ss = s.split(" ")
    Log.Message("The resulting array is: " + str(ss))
    Log.Message("There are " + str(len(ss)) + " words in the array")
    Log.Message("The first word is: " + ss[0])
    
def SplitDemo2():
    s = "Better late than never\r\nbut better never late.";
    # Split at line break character.
    ss = s.split("\r\n")
    Log.Message("The resulting array is: " + str(ss))
    Log.Message("There are " + str(len(ss)) + " words in the array")
    Log.Message("The first word is: " + ss[0])TestComplete has a similar method called aqString.GetListItem. It extracts a substring with the specified index from the input string. It was designed to read items from string lists. See Working With String Lists for more information. However, it also redefines delimiter characters and, like the split method, can be used to get sentences, separate words and so on.
Python
def SplitDemo3():
    s = "Better late than never but better never late."
    # Assign list separator to space character
    prevSep = aqString.ListSeparator
    aqString.ListSeparator = " "
    # Split by spaces
    Log.Message("There are " + str(aqString.GetListLength(s)) + " words in a string")
    Log.Message("The first word is: " + aqString.GetListItem(s,0))
    # Restore previous separator
    aqString.ListSeparator = prevSepRemoving Extra Spaces From a String
You may need to remove leading or trailing spaces, or both from a string you get in your tests. To do this, use Python's string.lstrip, string.rstrip and string.strip methods and the TestComplete aqString.Trim method.
Python
def TrimDemo():
  str="    Hallo    ";
  Log.Message("'" + aqString.Trim(str, aqString.stLeading) + "'") # Posts 'Hallo    '
  Log.Message("'" + aqString.Trim(str, aqString.stTrailing) + "'") # Posts '    Hallo' 
  Log.Message("'" + aqString.Trim(str, aqString.stAll) + "'") # Posts 'Hallo'You may also want to exclude extra inner whitespaces from a string. To do this, split the string into separate words and then put the words together separating them with one space character.
Replacing Characters and Substrings
You may need to find and replace a character or substring within a string. You can do this in two ways: by using the aqString.Replace method or by using the replace method of a string object.
The aqString object method is much easier to use. It can be used when you need to change certain character or string. It allows you to specify whether the search should be case-sensitive or not. Below is an example that demonstrates how to use this method:
Python
def StringReplaceDemo():
  str = "Hi, Bob. Have you seen Bob Robbinson?"
  str = aqString.Replace(str, "Bob", "Jack", True)
  Log.Message(str)The String.replace method is more complicated, but it offers more flexibility. You can change not only a character or string, but also all the fragments matching the specified regular expression pattern.
The first sample demonstrates how to change a string using the replace method.
Python
import re
def RegExReplaceDemo1():
  str = "Hi, Bob. Have you seen Bob Robbinson?"
  # Define regular expression pattern
  regexp = "Bob"
  # Perform replace operation
  str = re.sub(regexp, "Jack", str)
  Log.Message(str)The second example demonstrates how to replace a substring with alternative parts. The patterns of alternative parts are separated by pipe characters (" | "). For instance in the example below, the ht(ml|m) pattern matches both html and htm:
Python
def RegExReplaceDemo2():
  str = "The html is widely used in Internet. The HTM file is a text file with tags."
  # Define regular expression pattern.
  regexp = "(html|htm)"
  # Perform replace operation
  str = re.sub(regexp,"hypertext markup language", str, flags=re.I)
  Log.Message(str)Furthermore, using regular expressions, you can search for the text fragments matching the specified format. In the example below, all the dates written in the DD/MM/YYYY format are replaced with the Some Date string. This operation can be useful, for example, when comparing two reports that contain the date they were generated.
Changing the Letter Case
The TestComplete aqString object and the native string object have methods that convert uppercase letters to lowercase letters and vice versa. They are: aqString.ToLower, aqString.ToUpper, String.lower, String.upper and String.title.
The code below demonstrates how to use these methods.
Python
def LetterCaseDemo():
  str="The word 'Champagne' is of French origin"
  # Converting to lower case
  Log.Message(aqString.ToLower(str))
  Log.Message(str.lower())
  # Converting to upper case
  Log.Message(aqString.ToUpper(str))
  Log.Message(str.upper())
  # Converting to title case
  Log.Message(str.title())Working With String Lists
Some scripting objects, generally, controls like ListBoxes, ComboBoxes, Memos, etc., return information about their state or contents as string lists. The individual data elements (or items) in these lists are separated by commas, line breaks, carriage returns or some other delimiter characters.
The aqString object has a number of specific methods (AddListItem, ChangeListItem, DeleteListItem, GetListItem and GetListLength) that deal with such lists of strings. The AddListItem method adds an item to the list, the DeleteListItem method removes an item from the list, the GetListItem method retrieves the item with the given index, the ChangeListItem method assigns a new value to the given item and the GetListLength method returns the total number of items in the string list.
The symbol that is used as a separator of list items is defined by the ListSeparator property. By default, the list separator is a pipe character "|", but you can also assigned a comma, column, line break, carriage return, tab character or any other printable and non-printable character, or even several characters.
The sample code below demonstrates how you can work with string lists returned by scripting objects.
Python
def ListDialogOptions():
    # Get a string with dialog options
    OptStr = UserForms.UserForm1.SaveDialog1.Options
    # Assign list separator to comma
    prevSep = aqString.ListSeparator
    aqString.ListSeparator = ","
    # Get the number of dialog options 
    Log.Message("The dialog has " + str(aqString.GetListLength(OptStr)) + " option(s) enabled:")
    # Iterate through the options list
    for i  in range(0, aqString.GetListLength(OptStr)):
      #Get the option and post it to log
      Log.Message("Option No " + str((i+1)) + " is: "+ aqString.GetListItem(OptStr, i))
    # Restore previous separator
    aqString.ListSeparator = prevSep;
def ManageMemoText():
    # Get a string with memo lines
    StrList = UserForms.UserForm1.cxMemo1.Lines.Text
    # Post the memo contents to log
    Log.Message(UserForms.UserForm1.cxMemo1.Lines.Text)
    # Assign list separator to newline character
    prevSep = aqString.ListSeparator
    aqString.ListSeparator = "\n"
    # Append one more line 
    StrList = aqString.AddListItem(StrList, "Last Line")
    Log.Message(str(StrList))
    # Change the value of the first line
    StrList = aqString.ChangeListItem(StrList, "New First Line",0)
    # Set the memo contents to a new list
    UserForms.UserForm1.cxMemo1.Lines.Text = StrList
    # Post the memo contents to log
    Log.Message(UserForms.UserForm1.cxMemo1.Lines.Text)
    # Restore previous separator
    aqString.ListSeparator = prevSep