! Copyright Jérémie Burgalat (2017)
!
! burgalat.jeremie@gmail.com
!
! This software is a computer program whose purpose is to provide configuration
! file and command line arguments parsing features to Fortran programs.
!
! This software is governed by the CeCILL-B license under French law and
! abiding by the rules of distribution of free software. You can use,
! modify and/ or redistribute the software under the terms of the CeCILL-B
! license as circulated by CEA, CNRS and INRIA at the following URL
! "http://www.cecill.info".
!
! As a counterpart to the access to the source code and rights to copy,
! modify and redistribute granted by the license, users are provided only
! with a limited warranty and the software's author, the holder of the
! economic rights, and the successive licensors have only limited
! liability.
!
! In this respect, the user's attention is drawn to the risks associated
! with loading, using, modifying and/or developing or reproducing the
! software by the user in light of its specific status of free software,
! that may mean that it is complicated to manipulate, and that also
! therefore means that it is reserved for developers and experienced
! professionals having in-depth computer knowledge. Users are therefore
! encouraged to load and test the software's suitability as regards their
! requirements in conditions enabling the security of their systems and/or
! data to be ensured and, more generally, to use and operate it in the
! same conditions as regards security.
!
! The fact that you are presently reading this means that you have had
! knowledge of the CeCILL-B license and that you accept its terms.
!! file: strings.F90
!! summary: Strings manipulation source file
!! author: burgalat
!! date: 2017
#include "defined.h"
MODULE STRINGS
!! Fortran strings manipulation module
!!
!! This module provides methods and objects to manipulate Fortran (allocatable) strings. It defines
!! a doubly linked-list of strings, [[strings(module):words(type)]] and several methods to format
!! strings or convert them in other intrinsic types.
USE ERRORS
IMPLICIT NONE
PRIVATE
PUBLIC :: str2dble_sc,str2dble_ve,str2real_sc,str2real_ve
! errors module (not used but propagated)
PUBLIC :: noerror,error, error_to_string,aborting
! misc module methods
PUBLIC :: to_string, from_string, string_is, remove_quotes, format_string, &
format_paragraph, strip_newline, tokenize, str_length, str_endswith, &
str_startswith, str_to_lower, str_to_upper, str_add_attributes, &
str_delete_attributes, str_reset_attributes, str_remove, str_replace
! words object related methods
PUBLIC :: words_length, words_insert, words_append, words_prepend, words_get, &
words_set, words_get_max_width, words_get_total_width, words_pop, &
words_remove, words_next, words_previous, words_reset, &
words_valid, words_current, words_extend, words_reverse, &
words_reversed, words_dump, words_to_string, words_to_vector, &
new_words, words_clear
! Operators
PUBLIC :: ASSIGNMENT(=), OPERATOR(/=), OPERATOR(==)
INTEGER, PUBLIC, PARAMETER :: st_string = 1 !! String type ID
INTEGER, PUBLIC, PARAMETER :: st_logical = 2 !! Logical type ID
INTEGER, PUBLIC, PARAMETER :: st_complex = 3 !! Complex type ID
INTEGER, PUBLIC, PARAMETER :: st_integer = 4 !! Integer type ID
INTEGER, PUBLIC, PARAMETER :: st_real = 5 !! Real type ID
!> List of types names
CHARACTER(len=*), DIMENSION(5), PARAMETER, PUBLIC :: st_type_names = &
(/ 'string ', 'logical', 'complex', 'integer', 'real '/)
INTEGER, PUBLIC, PARAMETER :: st_slen = SSLEN !! Maximum short string length
INTEGER, PUBLIC, PARAMETER :: st_llen = SLLEN !! Maximum long string length
INTEGER, PUBLIC, PARAMETER :: FC_BLACK = 30 !! Black foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_RED = 31 !! Red foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_GREEN = 32 !! Green foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_YELLOW = 33 !! Yellow foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_BLUE = 34 !! Blue foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_MAGENTA = 35 !! Magenta foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_CYAN = 36 !! Cyan foreground csi code
INTEGER, PUBLIC, PARAMETER :: FC_WHITE = 37 !! White foreground csi code
INTEGER, PUBLIC, PARAMETER :: BG_BLACK = 40 !! Black foreground csi code
INTEGER, PUBLIC, PARAMETER :: BG_RED = 41 !! Black background csi code
INTEGER, PUBLIC, PARAMETER :: BG_GREEN = 42 !! Green background csi code
INTEGER, PUBLIC, PARAMETER :: BG_YELLOW = 43 !! Yellow background csi code
INTEGER, PUBLIC, PARAMETER :: BG_BLUE = 44 !! Blue background csi code
INTEGER, PUBLIC, PARAMETER :: BG_MAGENTA = 45 !! Magenta background csi code
INTEGER, PUBLIC, PARAMETER :: BG_CYAN = 46 !! Cyan background csi code
INTEGER, PUBLIC, PARAMETER :: BG_WHITE = 47 !! White background csi code
INTEGER, PUBLIC, PARAMETER :: ST_NORMAL = 0 !! Normal (regular) attribute
INTEGER, PUBLIC, PARAMETER :: ST_BOLD = 1 !! Bold (brighter) attribute
INTEGER, PUBLIC, PARAMETER :: ST_ITALIC = 3 !! Italic attribute (sometimes reverse video or underline)
INTEGER, PUBLIC, PARAMETER :: ST_UNDERLINE = 4 !! Underline attribute
INTEGER, PUBLIC, PARAMETER :: ST_BLINK = 5 !! Slow blink mode
!> List of all attributes in a vector
INTEGER, PUBLIC, PARAMETER, DIMENSION(21) :: attributes = [FC_BLACK, &
FC_RED, &
FC_GREEN, &
FC_YELLOW, &
FC_BLUE, &
FC_MAGENTA, &
FC_CYAN, &
FC_WHITE, &
BG_BLACK, &
BG_RED, &
BG_GREEN, &
BG_YELLOW, &
BG_BLUE, &
BG_MAGENTA, &
BG_CYAN, &
BG_WHITE, &
ST_NORMAL, &
ST_BOLD, &
ST_ITALIC, &
ST_UNDERLINE, &
ST_BLINK &
]
!> [[words(type)]] object assignement interface
INTERFACE ASSIGNMENT(=)
MODULE PROCEDURE ws_affect
END INTERFACE
!> [[words(type)]] interface constructor
!!
!! The interface encapsulates the two constructors of the words object:
!!
!! - [[new_words_1(function)]] initializes the object with a single word.
!! - [[new_words_str(function)]] initializes the object with a string that is
!! splitted according to a given delimiter.
INTERFACE new_words
MODULE PROCEDURE new_words_1, new_words_str
END INTERFACE
!> Clear either a scalar or a vector of list of [[words(type)]]
!!
!! The interface encapsulates words _destructors_, that deallocate memory used
!! by the given list(s) of words. This method should be called anytime words
!! object(s) is no longer used to avoid memory leaks.
!! @note
!! If the library support Derived type finalization, calling destructor is not
!! mandatory.
INTERFACE words_clear
MODULE PROCEDURE ws_clear_sc, ws_clear_ve
END INTERFACE
!> Extend a given [[words(type)]] object either by another or by a string
!!
!! The interface encapsulates two subroutines:
!!
!! - [[ws_extend_ws(subroutine)]](this,other) which extends __this__ by __other__
!! (both are words objects).
!! - [[ws_extend_str(subroutine)]](this,str,delimiter,merge) which splits __str__
!! according to __delimiter__ (and optionally __merge__) and then extends
!! __this__ with the resulting tokens.
INTERFACE words_extend
MODULE PROCEDURE ws_extend_ws,ws_extend_str
END INTERFACE
!> Convert an intrinsic type value to a string
!!
!! This (very) generic interface provides conversion functions from
!! intrinsic types to ALLOCATED string.
!!
!! ```
!! (1) FUNCTION to_string(value) RESULT(str)
!! (2) FUNCTION to_string(value,fmt,width) RESULT(str)
!! ```
!! Where :
!!
!! - __value__ is the value to convert
!! - __fmt__ is a string the format descriptor of the output string. Surrounding
!! parenthesis can be omitted.
!! - __width__ is an integer with the width of the output string (which should be given
!! in __fmt__ anyway).
!! - __str__ is an allocatable string with the converted value in output, or an empty
!! string if the conversion failed.
INTERFACE to_string
MODULE PROCEDURE int2str_as,int2str_fs
MODULE PROCEDURE log2str_as,log2str_fs
MODULE PROCEDURE real2str_as,real2str_fs
MODULE PROCEDURE dble2str_as,dble2str_fs
MODULE PROCEDURE cplx2str_as,cplx2str_fs
END INTERFACE
!> Convert a string into an intrisinc type
!!
!! All methods defined in the interface are functions which take in arguments,
!! a string (input) and an output variable with the relevant type (or vectors of both).
!! They always return an error object which is set to -5 error code (i.e. cannot cast value)
!! on error, otherwise [[errors(module):noerror(variable)]].
INTERFACE from_string
MODULE PROCEDURE str2int_sc,str2log_sc,str2real_sc,str2dble_sc,str2cplx_sc
MODULE PROCEDURE str2int_ve,str2log_ve,str2real_ve,str2dble_ve,str2cplx_ve
END INTERFACE
!> Define a linked word
!!
!! Linked words are only intended to be used within a words type.
!! It's part of the doubly linked list words.
TYPE, PUBLIC :: word
#if HAVE_FTNDTSTR
CHARACTER(len=:), ALLOCATABLE :: value !! Value of the word
#else
!> Value of the word
!!
!! @warning
!! It is always limited to [[strings(module):st_slen(variable)]] characters.
CHARACTER(len=st_slen) :: value = ''
#endif
TYPE(word), PRIVATE, POINTER :: next => null() !! Next word in the list of words
TYPE(word), PRIVATE, POINTER :: prev => null() !! Previous word in the list of words
END TYPE word
!> Define a list of words
TYPE, PUBLIC :: words
INTEGER :: nw = 0 !! Number of word in the list
TYPE(word), PRIVATE, POINTER :: head => null() !! First word in the list
TYPE(word), PRIVATE, POINTER :: tail => null() !! Last word in the list
TYPE(word), PRIVATE, POINTER :: iter => null() !! Current word (iterator)
#if HAVE_FTNPROC
CONTAINS
PROCEDURE, PRIVATE :: ws_extend_ws
PROCEDURE, PRIVATE :: ws_extend_str
PROCEDURE, PUBLIC :: length => words_length
!! Get the number of words in the list
PROCEDURE, PUBLIC :: insert => words_insert
!! Insert a word at given index
PROCEDURE, PUBLIC :: append => words_append
!! Append a word at the end of the list
PROCEDURE, PUBLIC :: prepend => words_prepend
!! Prepend a word at the beginning of the list
PROCEDURE, PUBLIC :: get => words_get
!! Get the word at given index
PROCEDURE, PUBLIC :: set => words_set
!! Set a word at given index
PROCEDURE, PUBLIC :: max_width => words_get_max_width
!! Get the width of the biggest word in the list
PROCEDURE, PUBLIC :: total_width => words_get_total_width
!! Get the total width of the words stored in the list
PROCEDURE, PUBLIC :: reverse => words_reverse
!! Reverse the list in place
PROCEDURE, PUBLIC :: reversed => words_reversed
!! Get a reversed copy of the list
PROCEDURE, PUBLIC :: dump => words_dump
!! Dump words of the list (on per line)
PROCEDURE, PUBLIC :: tostring => words_to_string
!! Convert the list in a single string
PROCEDURE, PUBLIC :: to_vector => words_to_vector
!! Convert the list in a vector
PROCEDURE, PUBLIC :: pop => words_pop
!! Pop a word from the list and returns it
PROCEDURE, PUBLIC :: remove => words_remove
!! Remove a word from the list
PROCEDURE, PUBLIC :: next => words_next
!! Go to the next word in the list
PROCEDURE, PUBLIC :: previous => words_previous
!! Go to the previous word in the list
PROCEDURE, PUBLIC :: reset => words_reset
!! Reset the list's iterator
PROCEDURE, PUBLIC :: valid => words_valid
!! Check if iterator position is valid
PROCEDURE, PUBLIC :: current => words_current
!! Get the current word in the list
GENERIC, PUBLIC :: extend => ws_extend_ws,ws_extend_str
!! Extend a list with either a string or another list of words
#endif
END TYPE words
CONTAINS
FUNCTION word_length(this) RESULT(lgth)
!! Get the trimmed length of the word object
TYPE(word), INTENT(in) :: this
!! A word object
INTEGER :: lgth
!! The length of the word's value (without trailing spaces)
#if HAVE_FTNDTSTR
IF (.NOT.ALLOCATED(this%value)) THEN
lgth = 0 ; RETURN
ENDIF
#endif
lgth = LEN_TRIM(this%value)
RETURN
END FUNCTION word_length
SUBROUTINE disconnect_word(this)
!! Disconnect a word object
!!
!! The object is no more connected to its neighbours which are connected together.
!! @note
!! After this method is called the object is no longer connected to its parent words
!! object and should be deallocated in order to avoid memory leaks.
TYPE(word), INTENT(inout) :: this
!! A word object to disconnect
TYPE(word), POINTER :: pw,nw
nw => this%next ; pw => this%prev
IF (ASSOCIATED(nw)) nw%prev => pw
IF (ASSOCIATED(pw)) pw%next => nw
RETURN
END SUBROUTINE disconnect_word
FUNCTION new_words_1(value) RESULT(list)
!! words constructor
!!
!! The method initializes a list of words with a single value.
CHARACTER(len=*), INTENT(in), OPTIONAL :: value
!! An optional string with the value to set in the list
TYPE(words) :: list
!! A words object
IF (PRESENT(value)) CALL ini_word(list,value)
RETURN
END FUNCTION new_words_1
FUNCTION new_words_str(string,delimiter,merge,protect) RESULT(list)
!! words constructor (extended version)
!!
!! The method splits an input strings and initializes a words object with the resulting tokens.
CHARACTER(len=*), INTENT(in) :: string
!! A string used to initialize the words object
CHARACTER(len=*), INTENT(in) :: delimiter
!! A string with the words delimiters.
!!
!! Note that each character is seen as a single delimiter.
LOGICAL, INTENT(in), OPTIONAL :: merge
!! An optional boolean control flag that instructs the method
!! wether to merge or not successive delimiters (default to .false.)
LOGICAL, INTENT(in), OPTIONAL :: protect
!! An optional boolean flag with .true. (default) to indicate that delimiter characters
!! between quotes are protected
TYPE(words) :: list
!! A new words object
CHARACTER(len=:), ALLOCATABLE :: seps
LOGICAL :: zmerge, zprotect
zmerge = .false. ; zprotect = .true. ; seps = ' '
IF (PRESENT(merge)) zmerge = merge
IF (PRESENT(protect)) zprotect = protect
IF (LEN(delimiter) > 0) seps = delimiter
CALL ws_extend_str(list,string,seps,zmerge,zprotect)
RETURN
END FUNCTION new_words_str
SUBROUTINE ws_affect(this,other)
!! words object assignment operator subroutine
TYPE(words), INTENT(out) :: this
!! A words object to be assigned
TYPE(words), INTENT(in) :: other
!! A words object to assign
TYPE(word), POINTER :: cur
CALL ws_clear_sc(this)
IF (other%nw == 0) THEN
RETURN
ELSE
cur => other%head
DO WHILE(associated(cur))
#if HAVE_FTNDTSTR
IF (.NOT.ALLOCATED(cur%value)) THEN
CALL words_append(this,"")
ELSE
CALL words_append(this,cur%value)
ENDIF
#else
CALL words_append(this,cur%value)
#endif
IF (ASSOCIATED(cur,other%iter)) this%iter => this%tail
cur => cur%next
ENDDO
ENDIF
RETURN
END SUBROUTINE ws_affect
SUBROUTINE ini_word(this,value)
!! Initialize the first word of a list of words
!!
!! This subroutine is not a constructor. It is only intended to set the first word
!! object in a words object.
TYPE(words), INTENT(inout) :: this
!! A words object reference
CHARACTER(len=*), INTENT(in) :: value
!! A string with the word used to initialize the list
ALLOCATE(this%head)
this%head%next => null()
this%head%prev => null()
this%tail => this%head
ASSIGN_DTSTR(value,this%tail%value)
this%nw = 1
RETURN
END SUBROUTINE ini_word
SUBROUTINE ws_clear_sc(obj)
!! Clear a list of words
!!
!! This subroutine deallocates all memory used by the given words object.
!! @warning
!! The subroutine should be called whenever a words is no more used (e.g. at
!! the end of the current scope), otherwise memory leaks could occur.
TYPE(words),INTENT(inout), TARGET :: obj
!! A words object to clear
TYPE(word), POINTER :: cur,next
IF (obj%nw == 0) RETURN
cur => obj%head
DO WHILE(ASSOCIATED(cur))
next => cur%next
CALL disconnect_word(cur)
#if HAVE_FTNDTSTR
IF (ALLOCATED(cur%value)) DEALLOCATE(cur%value)
#endif
DEALLOCATE(cur)
cur => next
ENDDO
obj%nw = 0
obj%head => null() ; obj%tail => null()
obj%iter => null()
END SUBROUTINE ws_clear_sc
SUBROUTINE ws_clear_ve(objs)
!! Clear a vector of list of words
!!
!! This subroutine deallocates all memory used by the given vector of words objects.
!! @warning
!! The subroutine should be called whenever a words is no more used (e.g. at the end
!! of the current scope), otherwise memory leaks could occur.
TYPE(words),INTENT(inout), DIMENSION(:) :: objs
!! A vector of words objects to clear
TYPE(word), POINTER :: cur,next
INTEGER :: i
DO i=1,SIZE(objs)
IF (objs(i)%nw == 0) CYCLE
cur => objs(i)%head
DO WHILE(ASSOCIATED(cur))
next => cur%next
CALL disconnect_word(cur)
DEALLOCATE(cur)
cur => next
ENDDO
objs(i)%nw = 0
objs(i)%head => null() ; objs(i)%tail => null()
ENDDO
END SUBROUTINE ws_clear_ve
SUBROUTINE ws_extend_ws(this, other)
!! Extend a list of words with another one
OBJECT(words), INTENT(inout) :: this
!! A words object to extend
TYPE(words), INTENT(in) :: other
!! A words object to extend with
TYPE(word), POINTER :: cw
IF (other%nw == 0) RETURN
cw => other%head
DO WHILE(ASSOCIATED(cw))
CALL words_append(this,cw%value) ; cw => cw%next
ENDDO
RETURN
END SUBROUTINE ws_extend_ws
SUBROUTINE ws_extend_str(this,str,delimiter,merge,protect)
!> Extend a list of word with a given string
!! @details The method adds a new list of words to the current list by
!! splitting a string using a set of delimiters.
!!
!! - If __delimiter__ is not given, THEN blank space is used.
!! - __delimiter__ can be a string of any length, but each character of
!! the sequence is seen as a single delimiter. Each time one of these
!! special character is seen on the string, it is splitted.
!! - If __protect__ is set to .true. THEN delimiter enclosed by
!! either single or double quotes are protected.
!! - The optional argument __merge__ instructs the method wether to merge
!! or not successive delimiters in the string.
!!
!! For example, considering the following string:
!!
@verbatim "I like coffee and bananas." @endverbatim
!! - Used with only __delimiter__ = " e", the method returns the list:
!! "I","lik","","coff","","","and","bananas"
!! - Used with both __delimiter__ = " e" and __merge__ = .true. :
!! "I","lik","coff","and","bananas"
!! @warning
!! The method does not trim or adjust the input string. Consequently, it can
!! add several empty words at the end of the list if the string is not well
!! defined.
!! @warning To avoid such problems, consider using TRIM() and ADJUSTL()
!! function on __str__ actual argument when calling this subroutine.
OBJECT(words), INTENT(inout), TARGET :: this
!! A words object to extend
CHARACTER(len=*), INTENT(in) :: str
!! A string to split in words
CHARACTER(len=*), INTENT(in), OPTIONAL :: delimiter
!! An optional string with the words delimiters (default to blank space).
LOGICAL, INTENT(in), OPTIONAL :: merge
!! An optional boolean control flag that instructs the method
!! wether to merge or not successive delimiters (default to .false.)
LOGICAL, INTENT(in), OPTIONAL :: protect
!! An optional boolean flag with .true. to indicate that
!! delimiter characters between quotes are protected
! - LOCAL
INTEGER :: sl,p,i,j,stat
LOGICAL :: zmerge,zprotect,indq,insq,outer
CHARACTER(len=:), ALLOCATABLE :: seps
CHARACTER(len=:), ALLOCATABLE :: curw
CHARACTER(len=1), PARAMETER :: sq = CHAR(39) ! single quote ascii code
CHARACTER(len=1), PARAMETER :: dq = CHAR(34) ! double quotes ascii code
stat=0 ; p=1 ; indq = .false. ; insq = .false.
seps = ' '
zmerge = .false. ; IF (PRESENT(merge)) zmerge = merge
zprotect = .true. ; IF (PRESENT(protect)) zprotect = protect
IF (PRESENT(delimiter)) THEN
IF (LEN(delimiter) > 0) seps = delimiter
ENDIF
sl = LEN(str) ; IF (sl == 0) RETURN
outer = (INDEX(str,sq) == 1 .AND. INDEX(str,sq,.true.) == LEN(str)) &
.OR.(INDEX(str,dq) == 1 .AND. INDEX(str,dq,.true.) == LEN(str))
! no delimiter found or (have outer quotes and should protect)
IF (SCAN(str,seps) == 0.OR.(outer.AND.zprotect)) THEN
CALL words_append(this,remove_quotes(str))
RETURN
ENDIF
! We have to loop...
i = 1 ; curw=''
DO
IF (i > sl) EXIT
p = SCAN(str(i:),seps)
IF (p == 0) THEN
! a gerer
curw = curw//TRIM(str(i:))
CALL words_append(this,TRIM(str(i:))) ; EXIT
curw=''
ELSE
IF (zprotect) THEN
j=i
! starting state
DO WHILE(j null()
IF (idx < 1 .OR. idx > words_length(this)) THEN
RETURN
ENDIF
IF (idx > (this%nw+1)/2) THEN
pted => this%tail
DO i=1,this%nw - idx ; pted => pted%prev ; ENDDO
ELSE
pted => this%head
DO i=1,idx-1 ; pted => pted%next ; ENDDO
ENDIF
RETURN
END FUNCTION ws_get_ptr
FUNCTION words_length(this) RESULT(res)
!! Get the size of the words object.
!!
!! The method returns the number of words stored in the given list of words.
OBJECT(words), INTENT(in) :: this !! A words object.
INTEGER :: res !! The number of words in the object.
res = this%nw
RETURN
END FUNCTION words_length
SUBROUTINE words_insert(this, idx, value)
!! Insert a word before given index in a list of words.
!!
!! The method inserts a new word before the given index in the list of words. If the given index is out
!! of range, the method prepends/appends the object based on the index value.
OBJECT(words), INTENT(inout) :: this
!! A words object.
INTEGER, INTENT(in) :: idx
!! An integer with the index of an object in the list. The new object will be inserted before that index.
CHARACTER(len=*), INTENT(in) :: value
!! A string with the word to insert in the list.
TYPE(word), POINTER :: welt,nx,pv
INTEGER :: i
welt => null() ; nx => null() ; pv => null()
IF (this%nw == 0) THEN
CALL ini_word(this,value)
ELSE IF (idx > this%nw) THEN
this%nw = this%nw + 1
welt => this%tail
allocate(this%tail)
ASSIGN_DTSTR(value,this%tail%value)
this%tail%prev => welt
this%tail%prev%next => this%tail
ELSE IF (idx <= 1) THEN
this%nw = this%nw + 1
welt => this%head
allocate(this%head)
ASSIGN_DTSTR(value,this%head%value)
this%head%next => welt
this%head%next%prev => this%head
ELSE
IF (idx > (this%nw+1)/2) THEN
nx => this%tail
DO i=1, this%nw - idx ; nx => nx%prev ; ENDDO
ELSE
nx => this%head
DO i=1, idx-1 ; nx => nx%next ; ENDDO
ENDIF
pv => nx%prev
allocate(welt)
ASSIGN_DTSTR(value,welt%value)
welt%prev => pv ; welt%next => nx
pv%next => welt ; nx%prev => welt
this%nw = this%nw + 1
ENDIF
RETURN
END SUBROUTINE words_insert
SUBROUTINE words_append(this,value)
!! Append a word to the list of word
!!
!! The method appends a word to the list of word. This is a convinient wrapper to
!! [[strings(module)::words_insert(subroutine)] to add a new word at the beginning of the list.
OBJECT(words), INTENT(inout) :: this !! A words object
CHARACTER(len=*), INTENT(in) :: value !! A string to append
CALL words_insert(this,this%nw+1,value)
RETURN
END SUBROUTINE words_append
SUBROUTINE words_prepend(this,value)
!! Prepend a word to the list of word
!!
!! The method prepends a word to the list of word. This is a convinient wrapper to
!! [[strings(module)::words_insert(subroutine)]] to add a new word at the end of the list.
OBJECT(words), INTENT(inout) :: this !! A words object
CHARACTER(len=*), INTENT(in) :: value !! A string to prepend
CALL words_insert(this,0,value)
RETURN
END SUBROUTINE words_prepend
FUNCTION words_get(this,idx,case) RESULT (res)
!! Get the word's value at given index
!!
!! The method attempts to get the word's value at the given index. If index is out of range
!! an empty string is returned.
!! @note
!! The returned string is always trimmed.
OBJECT(words), INTENT(in) :: this
!! A words object reference
INTEGER, INTENT(in) :: idx
!! An integer with the index of a word in the list
CHARACTER(len=5), INTENT(in), OPTIONAL :: case
!! An optional string with either 'upper' or 'lower' to get the value converted in the relevant case
CHARACTER(len=:), ALLOCATABLE :: res
!! The value of the word stored at given index in the list of words
TYPE(word), POINTER :: cur
cur => ws_get_ptr(this,idx)
IF (.not.associated(cur)) THEN
res = '' ; RETURN
ENDIF
IF (PRESENT(case)) THEN
IF (case == "upper") res = str_to_upper(cur%value)
IF (case == "lower") res = str_to_lower(cur%value)
ELSE
res = TRIM(cur%value)
ENDIF
RETURN
END FUNCTION words_get
SUBROUTINE words_set(this,idx,value)
!! Set a new value to a word object in the list of words at given index
!!
!! The method sets a new word at given index. If index is out of range, the method simply does nothing.
OBJECT(words), INTENT(inout) :: this !! A words object
INTEGER, INTENT(in) :: idx !! An integer with the index of the word object to modify in the list
CHARACTER(len=*), INTENT(in) :: value !! A string with the new value to set
TYPE(word), POINTER :: cur
cur => ws_get_ptr(this,idx)
IF (.NOT.ASSOCIATED(cur)) RETURN
cur%value = value
END SUBROUTINE words_set
FUNCTION words_get_max_width(this) RESULT(res)
!! Get the longest word's width in the words object
!!
!! The method computes and returns the longest (trimmed) word's width in the words object.
OBJECT(words), INTENT(in) :: this !! A words object
INTEGER :: res !! An integer with the maximum width (0 if the list is empty)
TYPE(word), POINTER :: cur
res = 0
IF (this%nw == 0) RETURN
cur => this%head ; res = word_length(cur)
DO WHILE(ASSOCIATED(cur%next))
cur => cur%next
IF (word_length(cur) > res) res = word_length(cur)
ENDDO
RETURN
END FUNCTION words_get_max_width
FUNCTION words_get_total_width(this) RESULT(width)
!! Get the total width of all words stored in the list of words
!!
!! The method computes and returns the total width of all words stored in
!! the list of words.
!! @note
!! Total width is computed using strings::word_length so it only takes
!! into account trimmed words (without trailing blanks)
!! @note
!! If csi codes have been added to words elements they are counted in the width.
OBJECT(words), INTENT(in) :: this !! A words object
INTEGER :: width !! Total length of the list of words
TYPE(word), POINTER :: cur
width = 0
IF (this%nw == 0) RETURN
cur => this%head ; width = word_length(cur)
DO WHILE(ASSOCIATED(cur%next))
cur => cur%next
width = width + word_length(cur)
ENDDO
cur => null()
RETURN
END FUNCTION words_get_total_width
SUBROUTINE words_reverse(this)
!! Reverse the list of words in-place
OBJECT(words), INTENT(inout) :: this
!! A words object to reverse
TYPE(word), POINTER :: loop,iwc,iwp
IF (this%nw <= 1) RETURN
loop => this%head ; iwc=> this%head ; iwp=> null()
DO WHILE(ASSOCIATED(loop%next))
loop => loop%next
iwp => iwc%prev ; iwc%prev => iwc%next ; iwc%next => iwp
iwc => loop
ENDDO
iwp=>this%tail%prev ; this%tail%prev=>this%tail%next ; this%tail%next=>iwp
iwc => this%head ; this%head => this%tail ; this%tail => iwc
loop => null() ; iwc => null() ; iwp => null()
RETURN
END SUBROUTINE words_reverse
FUNCTION words_reversed(this) RESULT(res)
!! Get a reversed copy of the list of words
OBJECT(words), INTENT(in) :: this
!! A words object to reverse
TYPE(words) :: res
!! A reversed copy of the input list of words
TYPE(word),POINTER :: cur
IF(this%nw == 0) RETURN
cur => this%tail
DO WHILE(ASSOCIATED(cur))
CALL words_append(res,cur%value)
IF (ASSOCIATED(cur,this%iter)) res%iter => res%tail
cur => cur%prev
ENDDO
cur => null()
RETURN
END FUNCTION words_reversed
SUBROUTINE words_dump(this,lun)
!! Dump the list of words
!!
!! The method dumps on the given logical unit the elements of the list one by line.
OBJECT(words), INTENT(in) :: this
!! A words object to dump
INTEGER, INTENT(in), OPTIONAL :: lun
!! An optional integer with the printing logical unit. If not given, the list is dumped on
!! standard output stream.
TYPE(word), POINTER :: cur
INTEGER :: lu
IF (this%nw == 0) RETURN
lu=6 ; IF (PRESENT(lun)) lu = lun
cur => this%head
DO WHILE(ASSOCIATED(cur))
WRITE(lu,'(a)') TRIM(cur%value)
cur => cur%next
ENDDO
cur => null()
RETURN
END SUBROUTINE words_dump
FUNCTION words_to_string(this, delimiter) RESULT(str)
!! Convert the list of words into a string
!!
!! The method converts the list of words into a string. In output, string is always
!! allocated even if the list is empty.
OBJECT(words), INTENT(in) :: this
!! A words object
CHARACTER(len=*), INTENT(in), OPTIONAL :: delimiter
!! An optional string used as delimiter between each words
CHARACTER(len=:), ALLOCATABLE :: str
!! An allocatable string with the list of words joined by the given delimiter (if any)
TYPE(word), POINTER :: cur
str = ''
IF (this%nw == 0) RETURN
cur => this%head
DO WHILE(ASSOCIATED(cur))
str=str//TRIM(cur%value)
IF (PRESENT(delimiter).AND..NOT.ASSOCIATED(cur,this%tail)) &
str=str//delimiter
cur => cur%next
ENDDO
RETURN
END FUNCTION words_to_string
FUNCTION words_to_vector(this,ret) RESULT(ok)
!! Convert the list of words into a vector of strings
!!
!! The method attempts to convert the list of words in a vector of strings.
!! If _this_ list of words is empty, the output vector is allocated with 0 elements and the method returns
!! .false., otherwise it returns .true.
!! @note
!! If elements in __this__ words object are wider than [[strings(module):st_slen(variable)]], output
!! values will be truncated.
OBJECT(words), INTENT(in) :: this
!! A words object reference
CHARACTER(len=st_slen), INTENT(out), ALLOCATABLE, DIMENSION(:) :: ret
!! An allocatable vector of assumed length string with the words of __this__
LOGICAL :: ok
!! Return status.
INTEGER :: l,mw
TYPE(word), POINTER :: iw
ok = .true.
l = words_length(this)
IF (l == 0) THEN
ALLOCATE(ret(0))
ok = .false.
RETURN
ENDIF
ALLOCATE(ret(l)) ; mw = LEN(ret(l))
ret(1:l) = ' ' ! really needed ?
iw => this%head ; l=1
DO WHILE(ASSOCIATED(iw))
ret(l) = TRIM(iw%value) ; l=l+1 ; iw => iw%next
ENDDO
END FUNCTION words_to_vector
FUNCTION words_pop(this,idx,move_forward) RESULT(value)
!! Pop a word in the list of words
!!
!! The method removes the word of the list at given index and returns it. If no index is given,
!! last word of the list is removed.
!!
!! If the index is out of range, the method does nothing and returns an empty string.
!!
!! By default, if the iterator is located on the item to be removed, it is moved backward before
!! deletion occurs. If __move\_forward__ is set to .true., the iterator is moved forward.
OBJECT(words), INTENT(inout) :: this
!! A words object
INTEGER, INTENT(in), OPTIONAL :: idx
!! Optional index of the word to delete
LOGICAL, INTENT(in), OPTIONAL :: move_forward
!! Move the iterator forward if needed. By default the iterator is moved backward.
CHARACTER(len=:), ALLOCATABLE :: value
!! The word's value at given index
LOGICAL :: zforward
INTEGER :: zidx
TYPE(word), POINTER :: cur
zidx=words_length(this) ; IF (PRESENT(idx)) zidx = idx
zforward = .false. ; IF (PRESENT(move_forward)) zforward = move_forward
cur => ws_get_ptr(this,zidx)
IF (.NOT.ASSOCIATED(cur)) THEN
value = '' ; RETURN
ELSE IF (ASSOCIATED(cur,this%iter)) THEN
IF (zforward) THEN
CALL words_next(this)
ELSE
CALL words_previous(this)
ENDIF
ENDIF
value = TRIM(cur%value)
CALL disconnect_word(cur)
DEALLOCATE(cur)
this%nw = this%nw - 1
RETURN
END FUNCTION words_pop
SUBROUTINE words_remove(this,idx,move_forward)
!! Remove the word of the list at given index
!!
!! The method removes the word of the list at given index. If no index is given, last word
!! of the list is removed.
!!
!! If the index is out of range, the method does nothing.
!!
!! By default, if the iterator is located on the item to be removed, it is moved backward before
!! deletion occurs. If __move\_forward__ is set to .true., the iterator is moved forward.
OBJECT(words), INTENT(inout) :: this
!! A words object
INTEGER, INTENT(in), OPTIONAL :: idx
!! Index of the word to delete
LOGICAL, INTENT(in), OPTIONAL :: move_forward
!! Move the iterator forward if needed. By default the iterator is moved backward.
LOGICAL :: zforward
INTEGER :: zidx
TYPE(word), POINTER :: cur
zidx=words_length(this) ; IF(PRESENT(idx)) zidx = idx
zforward = .false. ; IF (PRESENT(move_forward)) zforward = move_forward
cur => ws_get_ptr(this,idx)
IF (.NOT.ASSOCIATED(cur)) THEN
RETURN
ELSE IF (ASSOCIATED(cur,this%iter)) THEN
IF (zforward) THEN
CALL words_next(this)
ELSE
CALL words_previous(this)
ENDIF
ENDIF
CALL disconnect_word(cur)
DEALLOCATE(cur)
this%nw = this%nw - 1
RETURN
END SUBROUTINE words_remove
SUBROUTINE words_next(this)
!! Go to the next word in the list
OBJECT(words), INTENT(inout) :: this !! A words object
IF (ASSOCIATED(this%iter)) this%iter => this%iter%next
END SUBROUTINE words_next
SUBROUTINE words_previous(this)
!! Go to the previous word in the list
OBJECT(words), INTENT(inout) :: this !! A words object
IF (ASSOCIATED(this%iter)) this%iter => this%iter%prev
END SUBROUTINE words_previous
FUNCTION words_valid(this) RESULT(ret)
!! Check if the current iterated word is valid
OBJECT(words), INTENT(in) :: this !! A words object
LOGICAL :: ret !! A logical flag with .true. if the current iterated word is valid
ret = associated(this%iter)
END FUNCTION words_valid
FUNCTION words_current(this) RESULT(wrd)
!! Get current word value
OBJECT(words), INTENT(in) :: this
!! A words object
CHARACTER(len=:), ALLOCATABLE :: wrd
!! A string with the value of the current word or __an unallocated string__ if current word
!! is not valid (see [[strings(module):words_valid(function)]]).
IF (ASSOCIATED(this%iter)) THEN
wrd = this%iter%value
ENDIF
END FUNCTION words_current
SUBROUTINE words_reset(this,to_end)
!! Reset the iterator
!!
!! The method resets the iterator either at the beginning or at the end of the list of words
!! (if __to_end__ is set to .true.).
OBJECT(words), INTENT(inout) :: this !! A words object
LOGICAL, INTENT(in), OPTIONAL :: to_end !! An optional logical flag with .true. to reset the iterator at the end of the list
this%iter => this%head
IF (PRESENT(to_end)) THEN
IF (to_end) this%iter => this%tail
ENDIF
END SUBROUTINE words_reset
! Fancy string methods
! --------------------
FUNCTION tokenize(str,vector,delimiter,merge,protect) RESULT(ok)
!! Tokenize a string.
CHARACTER(len=*), INTENT(in) :: str
!! A string to tokenize
CHARACTER(len=*), INTENT(out), DIMENSION(:), ALLOCATABLE :: vector
!! An allocatable vector of strings with the tokens found. If string cannot be tokenized,
!! the vector is __allocated to 0 elements__ and the method returns .false..
CHARACTER(len=*), INTENT(in), OPTIONAL :: delimiter
!! An optional string with the words delimiters. It is set to blank space by default.
!! Note that each character is seen as a single delimiter.
LOGICAL, INTENT(in), OPTIONAL :: merge
!! An optional boolean control flag with .true. that instructs the method wether to
!! merge or not successive delimiters.
LOGICAL, INTENT(in), OPTIONAL :: protect
!! An optional boolean flag with .true. to indicate that delimiter characters between
!! quotes are protected.
LOGICAL :: ok
!! Return status (.true. on success)
CHARACTER(len=:), ALLOCATABLE :: seps
TYPE(words) :: tmp
LOGICAL :: zmerge,zprotect
zmerge = .false. ; zprotect = .true. ; seps = ' '
IF (PRESENT(merge)) zmerge = merge
IF (PRESENT(protect)) zprotect = protect
IF (PRESENT(delimiter)) THEN
IF (LEN(delimiter) > 0 ) seps = delimiter
ENDIF
tmp = new_words_str(str,seps,zmerge,zprotect)
ok = words_to_vector(tmp,vector)
CALL ws_clear_sc(tmp)
RETURN
END FUNCTION tokenize
FUNCTION remove_quotes(str) RESULT(ostr)
!! Strips outer quotes from string
!!
!! The function removes only external quotes from the input string
!! and returns the result in an allocatable string.
!! Quotes are removed only if they are the first and last non blank
!! characters. Either double and single quotes are stripped without distinction.
!! The output string is trimmed from leading and trailing blank spaces (after quotes removal !)
CHARACTER(len=*), INTENT(in) :: str !! A string to check
CHARACTER(len=:), ALLOCATABLE :: ostr !! A string without external quotes (if any).
CHARACTER(len=1), PARAMETER :: sq=CHAR(39), dq=CHAR(34)
CHARACTER(len=2), PARAMETER :: dsq=CHAR(39)//CHAR(34)
INTEGER :: i, j
IF (LEN_TRIM(str) == 0) RETURN
ostr = TRIM(ADJUSTL(str))
i = SCAN(ostr,sq//dq) ; j = SCAN(ostr,sq//dq,.true.)
IF (i == j) RETURN
IF (i /= 1) i = 0
IF (j /= LEN(ostr)) j = LEN(ostr)+1
ostr = TRIM(ostr(i+1:j-1))
RETURN
END FUNCTION remove_quotes
FUNCTION string_is(str) RESULT(ret)
!! Check if string represents an intrinsic type
!!
!! The method checks if the given string represents an intrinsic type. Both logical and complex type
!! are checked in a strict way :
!!
!! - A string is a logical if it is one of the following value: __.false.__, __.true.__, __F__, __T__.
!! - A string is potentially a complex if it has the following format: __(\*\*\*,\*\*\*)__ where
!! __\*\*\*__ is checked to see wether it is numerical or not.
!!
!! Valid numerical values can take the following forms:
!! ```
!! [0-9]
!! [0-9]*.?[0-9]*?([ed][+-]?[0-9]+)?
!! ```
!! Obviously if returned value is greater than 3, the string can be converted in
!! floating point value.
!!
!! Empty input string is simply considered to be of string type !
CHARACTER(len=*), INTENT(in) :: str
!! A string to check
INTEGER :: ret
!! An integer with the intrinsic type related to the string.
!!
!! Types are one of the following parameters
!!
!! - [[strings(module):st_string(variable)]] (1) for string
!! - [[strings(module):st_logical(variable)]] (2) for logical
!! - [[strings(module):st_complex(variable)]] (3) for complex
!! - [[strings(module):st_integer(variable)]] (4) for integer
!! - [[strings(module):st_real(variable)]] (5) for floating point value
CHARACTER(len=:), ALLOCATABLE :: zs,zzs
INTEGER :: j,l
ret = 1 ; IF (LEN_TRIM(str) == 0) RETURN
zs = str_to_lower(TRIM(ADJUSTL(str))) ; j = INDEX(zs,',') ; l = len(zs)
IF (zs(1:1)=='('.AND.zs(l:l) == ')'.AND.j==INDEX(zs,',')) THEN
IF (j == 2 .OR. j == l-1) RETURN
zzs = TRIM(ADJUSTL(zs(2:j-1))) ; IF (what_(zzs) < 3) RETURN
zzs = TRIM(ADJUSTL(zs(j+1:l-1))) ; ret = what_(zzs)
IF (ret > 3) THEN ; ret = 3 ; ELSE ; ret = 1 ; ENDIF
ELSE
ret = what_(zs)
ENDIF
CONTAINS
FUNCTION what_(s) RESULT(is)
!! Check if the given string is numerical, logical or a simple string
!! @note
!! Input string should be in lower case, otherwise, the method will give a a wrong result.
!! @warning
!! The test performed for logical checking is quite strict : A string is considered as logical
!! if and only if it is one of the following values : __.false.__, __.true.__, __F__, __T__.
CHARACTER(len=*), INTENT(in) :: s
!! A string to check
INTEGER :: is
!! An integer with : __1__ for string, __2__ for logical, __4__ for integer and __5__ for real
LOGICAL :: dec,fdot,fexp
INTEGER :: i
CHARACTER(len=24), PARAMETER :: aset='abcfghijklmnopqrstuvwxyz'
CHARACTER(len=10), PARAMETER :: iset='1234567890'
CHARACTER(len=2), PARAMETER :: dset='ed'
CHARACTER(len=2), PARAMETER :: sset='+-'
CHARACTER(len=7), PARAMETER :: slog(4) = (/'.true. ','.false.',&
't ','f '/)
is = -1 ; dec = .false. ; fdot = dec ; fexp = fdot
DO i = 1,LEN(s)
IF (i == 1) THEN
! string does not start by [+-\.\d]
IF (VERIFY(s(i:i),'.'//iset//sset) /= 0) THEN
is = 1 ; EXIT
ENDIF
! update control flag for decimal part
dec = s(i:i) == '.' ; fdot = dec
ELSE
! check if char is in [a-z]
IF(VERIFY(s(i:i),aset) == 0) THEN
dec=.false. ; is = 1 ; EXIT
ELSE IF (s(i:i) == '.') THEN
! check for dot in decimal/exponent part (==> not a number
IF (fdot.OR.fexp) THEN
dec = .false. ; is = 1 ; EXIT
ENDIF
ELSE IF (VERIFY(s(i:i),dset)==0) THEN
IF (fexp) THEN
dec = .false. ; is = 1 ; EXIT
ENDIF
ELSE IF (VERIFY(s(i:i),sset) == 0) THEN
IF (VERIFY(s(i-1:i-1),dset) /= 0) THEN
dec = .false. ; is = 1 ; EXIT
ENDIF
ENDIF
fdot = (fdot .OR. s(i:i) == '.')
fexp = (fexp .OR. VERIFY(s(i:i), dset) == 0)
ENDIF
ENDDO
! it is a string
IF (is == 1) THEN
! but have the format of a logical
IF (any(slog == s)) is = 2
ELSE
IF ((fexp.AND.SCAN(s(LEN(s):LEN(s)),dset) /= 0)) THEN
is = 1
ELSE
is = 4
IF (fdot.OR.fexp) is = 5
ENDIF
ENDIF
END FUNCTION what_
END FUNCTION string_is
FUNCTION format_string(str,idt1,idto) RESULT(output)
!! Format the given string
!!
!! This function only replaces all '\\n' escape sequence in the given string by NEW_LINE() character.
!! The output string is eventually indented if optional arguments are set.
!! @warning
!! __idto__ is relative to __idt1__ !
CHARACTER(len=*), INTENT(in) :: str !! The string to format
INTEGER, INTENT(in), OPTIONAL :: idt1, & !! An optional integer with the indentation level of the first output line (default to 0)
idto !! An optional integer with the indentation level of all other output lines (default to 0)
CHARACTER(len=:), ALLOCATABLE :: output !! An allocatable string with the output formatted string.
! - LOCAL
INTEGER :: i,c,ti,mx
CHARACTER(len=:), ALLOCATABLE :: idts
IF (LEN_TRIM(str) == 0) THEN
ALLOCATE(output,source='') ; RETURN
ENDIF
i=0 ; IF (PRESENT(idt1)) i = MAX(i,idt1)
ALLOCATE(CHARACTER(len=i) :: output)
IF (i > 0) output(1:i) = CHAR(32)
! i0 is relative to i1 and must be >= 0
IF (PRESENT(idto)) i = MAX(i+idto,0)
ALLOCATE(CHARACTER(len=i+1) :: idts)
idts(1:1) = NEW_LINE('A') ; IF (i>1) idts(2:) = CHAR(32)
! Builds output string
c=1 ; mx = LEN_TRIM(str)
i = INDEX(str(c:),'\n') ; ti = c+i-1
IF (i == 0) THEN
output=output//TRIM(str(ti+1:mx))
ELSE
output=output//TRIM(str(c:ti-1)) ; c=ti+2
DO
i = INDEX(str(c:),"\n") ; ti = c+i-1
IF (i == 0) THEN
output=output//TRIM(str(ti+1:mx)) ; c = mx+1
ELSE
output=output//idts//str(c:ti-1) ; c = ti+2
ENDIF
IF (c > mx) EXIT
ENDDO
ENDIF
! print a newline if we have \n at the end of the string
IF (INDEX(TRIM(str),'\n',.true.) == mx-1.AND.TRIM(str) /= '\n') &
output=output//idts(1:1)
END FUNCTION format_string
FUNCTION format_paragraph(str,width,idt1,idto) RESULT(output)
!! Split and format a string over several lines
!!
!! The function splits an input string in words so output lines fit (almost) in __width__ characters.
!! The method handles indentation level (defined as leading blank spaces). It also accounts for known
!! csi (see [[strings(module):attributes(variable)]].
!! @note
!! Words are considered indivisible and thus output lines can sometimes exceed the maximum width if
!! there is not enough space to put a word (with the associated indentation if given). The default
!! behavior in that case is to print the word in a new line (with the correct leading blank spaces).
!! @warning
!! If __width__, __idt1__ and/or __idto__ have inconsistent values (e.g. __width__ <= __idt1__), the
!! method still computes the paragraph, but each words will be set on a new line with the appropriate
!! indentation.
CHARACTER(len=*), INTENT(in) :: str !! string with the content to split
INTEGER, INTENT(in) :: width !! An positive integer with the maximum width of a line
INTEGER, INTENT(in), OPTIONAL :: idt1 !! An optional integer with the indentation level of the first output line
INTEGER, INTENT(in), OPTIONAL :: idto !! An optional integer with the indentation level of the other output lines
CHARACTER(len=:), ALLOCATABLE :: output !! An allocatable string with the output content
CHARACTER(len=:), ALLOCATABLE :: idts,zs
INTEGER :: l1,lo,zmx,zw,cc,j,jj,l
zw = abs(width) ; zs = strip_newline(str)
zmx = LEN_TRIM(zs)
IF (zmx == 0) THEN
ALLOCATE(output,source='') ; RETURN
ENDIF
l1=0 ; IF (PRESENT(idt1)) l1 = MAX(l1,idt1)
ALLOCATE(CHARACTER(len=l1) :: output)
IF (l1 > 0) output(1:l1) = CHAR(32)
lo=l1 ; IF (PRESENT(idto)) lo = MAX(l1+idto,0)
ALLOCATE(CHARACTER(len=lo+1) :: idts)
idts(1:1) = NEW_LINE('A') ; IF (lo>=1) idts(2:len(idts)) = CHAR(32)
! Prints a message if user is just stupid...
IF (lo+1 > zw .OR. l1+1 > zw) THEN
output = str ; RETURN
ENDIF
! check if can just return the string as is
IF (zmx + l1 <= zw) THEN
output=output//TRIM(zs) ; RETURN
ENDIF
j=1 ; jj=1+l1
DO
! Gets next blank in input string
cc = INDEX(TRIM(zs(j:)),CHAR(32))
! no more blank
! Gets total length of csi between zs(j:j+cc-1)
! this value will be substracted to each length test
IF (cc == 0) THEN
l = csis_length(zs(j:))
IF (jj-1+LEN_TRIM(zs(j:))-l > zw) THEN
output = output//idts
ENDIF
output=output//TRIM(zs(j:))
EXIT ! we are at the last word : we must exit the infinite loop !
ELSE
l = csis_length(zs(j:j+cc-1))
IF (cc+jj-1-l > zw) THEN
output=output//idts//zs(j:j+cc-1) ; jj = lo+1+cc+1 - l
ELSE
output=output//zs(j:j+cc-1) ; jj = jj + cc - l
ENDIF
ENDIF
j = j + cc
ENDDO
CONTAINS
FUNCTION csis_length(str) RESULT(value)
! - DUMMY
CHARACTER(len=*), INTENT(in) :: str
! - RESULT
INTEGER :: value
! - LOCAL
INTEGER :: jc,iesc,im
LOGICAL :: tcsi
value = 0
jc=1
DO
IF (jc>LEN(str)) EXIT
! search for escape
iesc = INDEX(str(jc:),CHAR(27))
IF (iesc == 0) EXIT
! search for m
im = INDEX(str(jc+iesc:),"m")
! no m in the string after ESC --> this could not be a csi
IF (im == 0) EXIT
! check if this is really a csi and updates length
tcsi = is_csi(str(jc+iesc-1:jc+iesc+im-1))
jc = jc + iesc
IF (tcsi) THEN
value=value+im+1
jc=jc+im
ENDIF
ENDDO
END FUNCTION csis_length
END FUNCTION format_paragraph
FUNCTION strip_newline(str,rpl) RESULT(stripped)
!! Replace newline escape sequences by spaces
!!
!! The function replaces newline (both '\\n' escape sequence and Fortran NEW_LINE() character) in the
!! given string and returns the resulting string.
CHARACTER(len=*), INTENT(in) :: str !! A string to process
CHARACTER(len=1), INTENT(in), OPTIONAL :: rpl !! A optional single character used as substitution of escape sequences (blank space by default)
CHARACTER(len=:), ALLOCATABLE :: stripped !! An allocatable string with all newline sequences replaced by blank space or __rpl__ if given
CHARACTER(len=1) :: zrp
INTEGER :: i, j, ns
zrp = CHAR(32) ; IF(PRESENT(rpl)) zrp = rpl
IF (str == NEW_LINE('A')) THEN
stripped = zrp ; RETURN
ENDIF
ns = LEN_TRIM(str)
IF (ns == 0) THEN
ALLOCATE(stripped,source='') ; RETURN
ENDIF
ALLOCATE(CHARACTER(len=ns) :: stripped) ; stripped(1:ns) = CHAR(32)
i=1 ; j=1
DO
IF (str(i:i) == NEW_LINE('A')) THEN
stripped(j:j) = zrp
ELSE IF (i < ns) THEN
IF (str(i:i+1) == "\n") THEN
stripped(j:j) = zrp ; i=i+1
ELSE
stripped(j:j) = str(i:i)
ENDIF
ELSE
stripped(j:j) = str(i:i)
ENDIF
j=j+1 ; i=i+1
IF (i > ns .OR. j > ns) EXIT
ENDDO
IF (j < ns) stripped = stripped(1:j)
RETURN
END FUNCTION strip_newline
FUNCTION str_length(str) RESULT(res)
!! Get the length of the string object
!!
!! The method computes the length of the string. It differs from LEN intrinsic function as
!! it does not account for extra-characters of csi codes.
CHARACTER(len=*), INTENT(in) :: str !! String to process
INTEGER :: res !! The actual length of string (i.e. does not account for csi codes)
CHARACTER(len=:), ALLOCATABLE :: tmp
res = 0
IF (LEN(str) /= 0) THEN
tmp = str_reset_attributes(str)
res = LEN(tmp)
DEALLOCATE(tmp)
ENDIF
RETURN
END FUNCTION str_length
FUNCTION str_to_lower(str1) RESULT(str)
!! Convert the string in lower case
!!
!! The method converts the input string in lower case and accounts for
!! possible csi codes in the string.
CHARACTER(len=*), INTENT(in) :: str1 !! Input string to convert
CHARACTER(len=:), ALLOCATABLE :: str !! A copy of the string in lower case
INTEGER :: i,ic
IF (LEN(str1) /= 0) THEN
str = str1
DO i = 1, len(str1)
ic = ichar(str1(i:i))
IF (ic >= 65 .AND. ic < 90) str(i:i) = char(ic + 32)
ENDDO
ELSE
str=''
ENDIF
END FUNCTION str_to_lower
FUNCTION str_to_upper(str1) RESULT(str)
!! Convert the string in upper case
!!
!! The method converts the input string in upper case and accounts for
!! possible csi codes in the string.
CHARACTER(len=*), INTENT(in) :: str1 !! Input string to convert
CHARACTER(len=:), ALLOCATABLE :: str !! A copy of the string in upper case
INTEGER :: j,i,ic,icsi,lcsi
IF (LEN(str1) > 0) THEN
str = str1
i = 1
DO
IF (i > LEN(str)) EXIT
icsi = str_index_of_csi(str(i:),lcsi)
IF (icsi == 0) THEN
! no more csi the end of string is upper case converted
DO j=i,LEN(str)
ic = ichar(str(j:j))
IF (ic >= 97 .AND. ic < 122) str(j:j) = char(ic-32)
ENDDO
RETURN
ELSE IF (icsi == 1) THEN
i = i + lcsi
ELSE IF (icsi > 1) THEN
! csi is not the first word: we convert in upper case until its
! position THEN copy the csi and get back in the loop
DO j=i,i+icsi-2
ic = ichar(str(j:j))
IF (ic >= 97 .AND. ic < 122) str(j:j) = char(ic-32)
ENDDO
i = i + icsi + lcsi-1
ENDIF
ENDDO
ELSE
str=''
ENDIF
END FUNCTION str_to_upper
FUNCTION str_remove(string,substring,back,all) RESULT(str)
!! Remove substring from current string
!!
!! The function removes the first occurence of __substring__ in __string__ or all
!! its occurences if __all__ is explicitly set to .true..
CHARACTER(len=*), INTENT(in) :: string !! A string to search in
CHARACTER(len=*), INTENT(in) :: substring !! A string to search and removes from __string__
LOGICAL, INTENT(in), OPTIONAL :: back, & !! An optional boolean flag with .true. to begin search at the end of the string
all !! An optional boolean flag with .true. to remove all occurences of __substring__
CHARACTER(len=:), ALLOCATABLE :: str !! An allocatable string with __substring__ occurence(s) removed
LOGICAL :: zb,za
INTEGER :: is,j,zboff
str=''
zb = .false. ; za = .false.
IF (PRESENT(back)) zb = back
IF (PRESENT(all)) za = all
IF (za) zb=.false.
zboff = 0 ; IF (zb) zboff = 1
IF (LEN(string) == 0) RETURN
j=1
DO
IF (j>LEN(string)) EXIT
! search for substring
is = INDEX(string(j:),substring,back=zb)
IF (is == 0) THEN
! substring is not found : we get the last part of the string and return
str = str//string(j:) ; RETURN
ELSE IF (is == 1) THEN
j = j + LEN(substring)
ELSE
! substring is not at the begin of the string : saves the string
str = str//string(j:j+is-2)
j = j + is+LEN(substring)-1
ENDIF
! if we only want to str_remove ONE occurence we exit if substring
! has been found
IF (.NOT.(is==0.OR.za)) EXIT
ENDDO
IF (j <= LEN(string).AND..NOT.zb) str=str//string(j:)
RETURN
END FUNCTION str_remove
FUNCTION str_replace(string,old,new,back,all) RESULT(str)
!! Replace substring from current string
!!
!! The function replaces the first occurence of __old__ in __string__ by
!! __new__ or all its occurence(s) if __all__ is explicitly set to .true..
CHARACTER(len=*), INTENT(in) :: string !! A string to search in
CHARACTER(len=*), INTENT(in) :: old, & !! A string to search and replace
new !! A string to substitute to __old__
LOGICAL, INTENT(in), OPTIONAL :: back, & !! An optional boolean flag with .true. to begin search at the end of the string
all !! An optional boolean flag with .true. to replace all occurences of __old__
CHARACTER(len=:), ALLOCATABLE :: str !! An allocatable string with occurence(s) of __old__ replaced by __new__
LOGICAL :: zb,za
INTEGER :: is,j
str=''
zb = .false. ; za = .false.
IF (PRESENT(back)) zb = back
IF (PRESENT(all)) za = all
IF (za) zb = .NOT.za
IF (LEN(string) == 0) RETURN
j=1
DO
IF (j>LEN(string)) EXIT
! search for "old"
is = INDEX(string(j:),old,back=zb)
IF (is == 0) THEN
! "old" is not found : we get the last part of the string and return
str = str//string(j:) ; RETURN
ELSE IF (is == 1) THEN
str = str//new
j = j + LEN(old)
ELSE
! "old" is not at the begin of the string : saves the string
str = str//string(j:j+is-2)//new
j = j + is + LEN(old) - 1
ENDIF
IF (.NOT.(is==0.OR.za)) EXIT
ENDDO
IF (j <= LEN(str)) str=str//string(j:)
RETURN
END FUNCTION str_replace
FUNCTION str_endswith(string,substring,icase) RESULT(ret)
!! Check if string ends by substring
CHARACTER(len=*), INTENT(in) :: string
!! @param[in] string A string to check
CHARACTER(len=*), INTENT(in) :: substring
!! A string to search in __string__
LOGICAL, INTENT(in), OPTIONAL :: icase
!! An optional boolean flag with .true. to perform insensitive case search
LOGICAL :: ret
!! .true. if __string__ ends by __substring__, .false. otherwise.
CHARACTER(len=:), ALLOCATABLE :: zthis,zstr
INTEGER :: idx
LOGICAL :: noc
ret = .false.
noc = .false. ; IF (PRESENT(icase)) noc = icase
IF (LEN(string) == 0 .OR. LEN(substring) == 0) RETURN
zthis = str_reset_attributes(string) ; zstr=str_reset_attributes(substring)
IF (noc) THEN
idx = INDEX(str_to_lower(zthis),str_to_lower(zstr),.true.)
ELSE
idx = INDEX(zthis,zstr,.true.)
ENDIF
IF (idx == 0.OR.idx+str_length(zstr)-1 /= str_length(zthis)) RETURN
ret=.true.
END FUNCTION str_endswith
FUNCTION str_startswith(string,substring,icase) RESULT(ret)
!! Check if string starts by substring
CHARACTER(len=*), INTENT(in) :: string
!! A string to check
CHARACTER(len=*), INTENT(in) :: substring
!! A string to search in __string__
LOGICAL, INTENT(in), OPTIONAL :: icase
!! An optional boolean flag with .true. to perform insensitive case search
LOGICAL :: ret
!! .true. if __string__ starts by __substring__, .false. otherwise.
CHARACTER(len=:), ALLOCATABLE :: zthis,zstr
INTEGER :: idx
LOGICAL :: noc
ret = .false.
noc = .false. ; IF (PRESENT(icase)) noc = icase
IF (LEN(string) == 0 .OR. LEN(substring) == 0) RETURN
zthis = str_reset_attributes(string) ; zstr=str_reset_attributes(substring)
IF (noc) THEN
idx = INDEX(str_to_lower(zthis),str_to_lower(zstr))
ELSE
idx = INDEX(zthis,zstr)
ENDIF
IF (idx /= 1) RETURN
ret=.true.
END FUNCTION str_startswith
! CSI related functions
! ---------------------
FUNCTION str_add_attributes(string,attrs) RESULT(str)
!! Set csi attributes to the given string object
!!
!! The function adds csi (ANSI escape sequences) to the given string and
!! returns a copy of it.
CHARACTER(len=*), INTENT(in) :: string
!! @param[in] string A string object reference
INTEGER, INTENT(in), DIMENSION(:) :: attrs
!! A vector of integers with the code to add. Each __attrs__ value should refers to one i
!! of [[strings(module):attributes(variable)]] values.
CHARACTER(len=:), ALLOCATABLE :: str
!! An allocatable string with new csi codes added.
INTEGER :: j,iesc,im
CHARACTER(len=:), ALLOCATABLE :: tmp,csi
CHARACTER(len=4), PARAMETER :: rcsi = CHAR(27)//"[0m"
str=''
! 1) Check for input string
IF (LEN(string) == 0) RETURN
! 2) Removes last [0m if any and initializes output string
! we must remove only the last [0m if any
IF (INDEX(string,rcsi,.true.) == LEN(string)-3) THEN
tmp = str_remove(string,rcsi,back=.true.)
ELSE
tmp = string
ENDIF
! 3) Add all the given csi preceded by [0m at the beginning of the string
! if it does not start by an ANSI sequence
IF (INDEX(tmp,CHAR(27)//"[") /= 1) &
tmp = str_add_to_csi(rcsi,attrs)//tmp
! Loops on new string and updates csi codes
j=1
DO
IF (j>LEN(tmp)) EXIT
! search for escape
iesc = INDEX(tmp(j:),CHAR(27))
IF (iesc == 0) THEN
! no more ESC : cat until end of input string and exit
str = str//tmp(j:) ; EXIT
ELSE IF (iesc > 1) THEN
! ESC is not first char: copy until ESC
str = str//tmp(j:j+iesc-2)
ENDIF
! search for m
im = INDEX(tmp(j+iesc:),"m")
! no m in the string after ESC --> copy string (INCLUDING ESC) and leave
IF (im == 0) THEN
str = str//tmp(j+iesc-1:)
RETURN
ENDIF
csi = tmp(j+iesc-1:j+iesc+im-1)
! we have a csi: we add new codes to it
IF (is_csi(csi)) THEN
csi = str_add_to_csi(csi,attrs)
ENDIF
str = str//csi
j = j + iesc + im
ENDDO
IF (INDEX(str,rcsi,.true.) /= LEN(str)-3) str = str//rcsi
RETURN
END FUNCTION str_add_attributes
FUNCTION str_delete_attributes(string,attrs) RESULT(str)
!! Remove attributes to the given string
!!
!! The function removes list of csi (ANSI escape sequences) from the given
!! string and returns a copy of it.
CHARACTER(len=*), INTENT(in) :: string
!! Input string
INTEGER, INTENT(in), DIMENSION(:) :: attrs
!! A vector of integers with the code to remove. Each __attrs__ value should
!! refers to one of [[strings(module):attributes(variable)]] values.
CHARACTER(len=:), ALLOCATABLE :: str
!! An allocatable string with csi codes from __list__ removed
LOGICAL :: ok
INTEGER :: j,iesc,im
CHARACTER(len=:), ALLOCATABLE :: tmp,csi,csis
CHARACTER(len=st_slen), DIMENSION(:), ALLOCATABLE :: tks
CHARACTER(len=4), PARAMETER :: rcsi = CHAR(27)//"[0m"
str=''
IF (LEN(string) == 0) RETURN
! remove last [0m if found at the end of the string
IF (INDEX(string,rcsi,.true.) == LEN(string)-3) THEN
tmp = str_remove(string,rcsi,back=.true.)
ELSE
tmp = string
ENDIF
! Loops on new string and updates csi codes
j=1 ; csis=""
DO
IF (j>LEN(tmp)) EXIT
! search for escape
iesc = INDEX(tmp(j:),CHAR(27))
IF (iesc == 0) THEN
! no more ESC : cat until end of input string and exit
str = str//tmp(j:) ; EXIT
ELSE IF (iesc > 1) THEN
! ESC is not first char: copy until ESC
str = str//tmp(j:j+iesc-2)
ENDIF
! search for m
im = INDEX(tmp(j+iesc:),"m")
! no m in the string after ESC --> copy string (INCLUDING ESC) and leave
IF (im == 0) THEN
str = str//tmp(j+iesc-1:)
RETURN
ENDIF
csi = tmp(j+iesc-1:j+iesc+im-1)
! we have a csi: we add new codes to it
IF (is_csi(csi)) THEN
csi = str_del_from_csi(csi,attrs)
ENDIF
csis=csis//csi//"|"
str = str//csi
j = j + iesc + im
ENDDO
! Add [0m at the end of string if not found
IF (INDEX(str,rcsi,.true.) /= LEN(str)-3) str = str//rcsi
! resets all attributes if we only have [0m in final list
ok = tokenize(csis(1:LEN(csis)-1),tks,"|")
IF (ALL(tks == rcsi)) str = str_reset_attributes(str)
DEALLOCATE(tks)
RETURN
END FUNCTION str_delete_attributes
FUNCTION str_reset_attributes(string) RESULT(str)
!! Reset all csi codes of the string
!!
!! The method removes __all__ the known escape sequences from the input string.
CHARACTER(len=*), INTENT(in) :: string
!! Input string
CHARACTER(len=:), ALLOCATABLE :: str
!! An allocatable string with the copy of input string stripped off csi codes.
INTEGER :: j,iesc,im
LOGICAL :: tcsi
str = ""
IF (LEN(string) == 0) RETURN
j=1
DO
IF (j>LEN(string)) EXIT
! search for escape
iesc = INDEX(string(j:),CHAR(27))
IF (iesc == 0) THEN
str = str//string(j:) ; EXIT
ENDIF
! search for m
im = INDEX(string(j+iesc:),"m")
! no m in the string after ESC --> copy string (INCLUDING ESC) and leave
IF (im == 0) THEN
str = str//string(j+iesc-1:)
RETURN
ENDIF
! csi includes everything between ESC and m (excluding them):
! to check for csi it should begin by [ and then be a list of integers
! separated by ;
tcsi = is_csi(string(j+iesc-1:j+iesc+im-1))
IF (iesc > 1) THEN
str = str//string(j:j+iesc-2)
ENDIF
j = j + iesc ; IF (tcsi) j=j+im
ENDDO
RETURN
END FUNCTION str_reset_attributes
FUNCTION is_csi(value) RESULT(yes)
!! Check if string is a known csi
!!
!! The function only check for known csi code which are defined in [[strings(module):attributes(variable)]].
CHARACTER(len=*), INTENT(in) :: value
!! A Fortran intrinsic string to check
LOGICAL :: yes
!! .true. if it is a known csi, .false. otherwise
LOGICAL :: ok
CHARACTER(len=:), ALLOCATABLE :: tmp
TYPE(words) :: wtks
CHARACTER(len=st_slen), DIMENSION(:), ALLOCATABLE :: stks
INTEGER, DIMENSION(:), ALLOCATABLE :: nums
INTEGER :: i
yes = .false.
IF (LEN(value) < 4) RETURN
tmp = value(3:len(value)-1)
wtks = new_words(tmp,";")
ok = words_to_vector(wtks,stks)
CALL ws_clear_sc(wtks)
IF (.NOT.ok) RETURN
! if we cannot convert strings to integers : it is not a csi
IF (.NOT.from_string(stks,nums)) RETURN
DEALLOCATE(stks)
DO i=1, SIZE(nums)
IF (.NOT.ANY(attributes == nums(i))) RETURN
ENDDO
yes = .true.
END FUNCTION is_csi
FUNCTION str_add_to_csi(csi,list) RESULT(ncsi)
!! Add a new list of codes to the input csi string
!!
!! The method adds all the csi codes given in __list__ that are known by the module and not
!! already present in the input csi.
CHARACTER(len=*), INTENT(in) :: csi
!! A string with the input csi. It __must__ begin with "[" and ends with "m".
INTEGER, INTENT(in), DIMENSION(:) :: list
!! A vector of integers with the csi code to add. Each value of __list__ should be one of
!! [[strings(module):attributes(variable)]] values. All unknwon values are filtered out as well
!! as csi code already present in input __csi__.
CHARACTER(len=:), ALLOCATABLE :: ncsi
!! A new csi string or the input __csi__ if some "errors" occured (the input csi could not
!! be tokenized or none of __list__ values are left after filtering).
LOGICAL :: ok
CHARACTER(len=LEN(csi)), DIMENSION(:), ALLOCATABLE :: tks
CHARACTER(len=:), ALLOCATABLE :: tmp
INTEGER, DIMENSION(:), ALLOCATABLE :: zlist,nums
INTEGER :: i,j,ni,no
! 1) Filter input list :
! 1.1) Gets the list of current csi codes
ncsi = csi(3:len(csi)-1)
ok = tokenize(ncsi,tks,"; ",merge=.true.)
IF (.NOT.from_string(tks,nums)) THEN
ncsi = csi
RETURN
ENDIF
DEALLOCATE(tks)
! 1.2) Filter input list of new flags to add
! counts number of valid flags
j=0
DO i=1,SIZE(list)
! new flags must be in attributes but NOT in nums
IF (ANY(attributes==list(i).AND..NOT.ANY(nums == list(i)))) j=j+1
ENDDO
! No "valid" flags -> returns old csi
IF (j == 0) THEN ; ncsi = csi ; RETURN ; ENDIF
ni = SIZE(nums) ; no = j + ni
ALLOCATE(zlist(no)) ; zlist(1:ni) = nums(:) ; j = ni
DO i=1,SIZE(list)
! new flags must be in attributes but NOT in nums
IF (ANY(attributes==list(i).AND..NOT.ANY(nums == list(i)))) THEN
j=j+1 ; zlist(j) = list(i)
ENDIF
ENDDO
DEALLOCATE(nums)
! 2) Builds new csi
! Here we explictly set the first flag to 0 (i.e. reset attributes)...
ncsi = CHAR(27)//"[0;"
DO i=1,no
! ... So we get rid of all "0" flag in the list
IF (zlist(i) /= 0) THEN
tmp = to_string(zlist(i))
IF (LEN_TRIM(tmp) == 0) THEN
ncsi = csi ; RETURN
ENDIF
ncsi = ncsi//tmp
IF (i /= no) ncsi = ncsi//";"
ENDIF
ENDDO
ncsi = ncsi//"m"
END FUNCTION str_add_to_csi
FUNCTION str_del_from_csi(csi,list) RESULT(ncsi)
!! Remove a list of codes from the input csi string
!!
!! The method removes all the csi codes given in __list__ that are known by the
!! module and already present in the input csi.
CHARACTER(len=*), INTENT(in) :: csi
!! An intrinsic Fortran string with the input csi. It __must__ begin with "[" and ends with "m".
INTEGER, INTENT(in), DIMENSION(:) :: list
!! A vector of integers with the csi code to remove. Each value of __list__ should be one of
!! [[strings(module):attributes(variable)]] values. All unknwon values are filtered out.
CHARACTER(len=:), ALLOCATABLE :: ncsi
!! A new csi string or the input __csi__ if some "errors" occured (the input csi could not
!! be tokenized or none of __list__ values are left after filtering).
LOGICAL :: ok
CHARACTER(len=LEN(csi)), DIMENSION(:), ALLOCATABLE :: tks
CHARACTER(len=:), ALLOCATABLE :: tmp
INTEGER, DIMENSION(:), ALLOCATABLE :: nums
INTEGER :: i
ncsi = csi(3:len(csi)-1)
ok = tokenize(ncsi,tks,"; ",merge=.true.)
IF (.NOT.from_string(tks,nums)) THEN
ncsi = csi
RETURN
ENDIF
DEALLOCATE(tks)
tmp=""
DO i=1, SIZE(nums)
IF (ALL(nums(i) /= list).AND.nums(i) /= 0) THEN
! no need to check for to_string status : it is always ok !
tmp=tmp//to_string(nums(i))//";"
ENDIF
ENDDO
IF (LEN_TRIM(tmp) /= 0) THEN
ncsi=CHAR(27)//"[0;"//tmp(1:LEN(tmp)-1)//"m"
ELSE
ncsi=CHAR(27)//"[0m"
ENDIF
END FUNCTION str_del_from_csi
FUNCTION str_index_of_csi(str,length) RESULT(pos)
!! Get the position of the first known csi in string
!!
!! The method searches for the first known csi in string. The csi must contain known codes
!! (i.e. values of [[strings(module):attributes(variable)]]).
CHARACTER(len=*), INTENT(in) :: str !! A string to search in
INTEGER, INTENT(out) :: length !! Length of the csi in the string
INTEGER :: pos !! Position of the first csi found. It is set to 0 if no csi has been found.
INTEGER :: iesc,im
pos = 0 ; length = 0
! we need at least 4 chars to create a csi
IF (LEN_TRIM(str) < 4) RETURN
iesc = INDEX(str,CHAR(27))
IF (iesc == 0) RETURN
! search for m
im = INDEX(str(iesc:),"m")
! no m in the string after ESC --> copy string (INCLUDING ESC) and leave
IF (im == 0) RETURN
IF (.NOT.is_csi(str(iesc:iesc+im-1))) RETURN
pos = iesc ; length = im
END FUNCTION str_index_of_csi
! String conversion functions
! ---------------------------
FUNCTION str2int_sc(str, value) RESULT(ret)
!! Convert string value to integer value (scalar)
CHARACTER(len=*), INTENT(in) :: str !! String to convert
INTEGER, INTENT(out) :: value !! Output value
LOGICAL :: ret !! Return status (.true. on success)
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; zs = remove_quotes(str)
IF (string_is(zs) /= st_integer) THEN
ret = .false.
ELSE
READ(zs, *) value
ENDIF
RETURN
END FUNCTION str2int_sc
FUNCTION str2log_sc(str, value) RESULT(ret)
!! Convert string value to logical value (scalar)
CHARACTER(len=*), INTENT(in) :: str !! String to convert
LOGICAL, INTENT(out) :: value !! Output value
LOGICAL :: ret !! Return status (.true. on success)
CHARACTER(len=:), ALLOCATABLE :: zs
integer :: r
ret = .true. ; zs = remove_quotes(str)
r = string_is(zs)
IF (string_is(zs) /= st_logical) THEN
ret = .false.
ELSE
READ(zs, *) value
ENDIF
RETURN
END FUNCTION str2log_sc
FUNCTION str2real_sc(str, value) RESULT(ret)
!! Convert string value to simple precision floating precision value (scalar)
CHARACTER(len=*), INTENT(in) :: str !! String to convert
REAL(kind=4), INTENT(out) :: value !! Output value
LOGICAL :: ret !! Return status (.true. on success)
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true.; zs = remove_quotes(str)
IF (string_is(zs) < st_integer) THEN
ret = .false.
ELSE
READ(zs, *) value
ENDIF
RETURN
END FUNCTION str2real_sc
FUNCTION str2dble_sc(str, value) RESULT(ret)
!! Convert string value to double precision floating precision value (scalar)
CHARACTER(len=*), INTENT(in) :: str !! String to convert
REAL(kind=8), INTENT(out) :: value !! Output value
LOGICAL :: ret !! Return status (.true. on success)
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; zs = remove_quotes(str)
IF (string_is(zs) < st_integer) THEN
ret = .false.
ELSE
READ(zs, *) value
ENDIF
RETURN
END FUNCTION str2dble_sc
FUNCTION str2cplx_sc(str, value) RESULT(ret)
!! Convert string value to complex value (scalar)
CHARACTER(len=*), INTENT(in) :: str !! String to convert
COMPLEX(kind=4), INTENT(out) :: value !! Output value
LOGICAL :: ret !! Return status (.true. on success)
! - LOCAL
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; zs = remove_quotes(str)
IF (string_is(zs) /= st_complex) THEN
ret = .false.
ELSE
READ(zs, *) value
ENDIF
RETURN
END FUNCTION str2cplx_sc
FUNCTION str2int_ve(str, value) RESULT(ret)
!! Convert strings values to integer values (vector)
CHARACTER(len=*), INTENT(in), DIMENSION(:) :: str !! Vector of strings to convert
INTEGER, INTENT(out), DIMENSION(:), ALLOCATABLE :: value !! Vector of output values
LOGICAL :: ret !! Return status (.true. on success)
INTEGER :: i,ns
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; ns = SIZE(str) ; ALLOCATE(value(ns))
DO i=1,ns
zs = remove_quotes(str(i))
IF (string_is(zs) /= st_integer) THEN
ret = .false. ; DEALLOCATE(value) ; RETURN
ELSE
READ(zs, *) value(i)
ENDIF
ENDDO
RETURN
END FUNCTION str2int_ve
FUNCTION str2log_ve(str, value) RESULT(ret)
!! Convert strings values to logical values (vector)
CHARACTER(len=*), INTENT(in), DIMENSION(:) :: str !! Vector of strings to convert
LOGICAL, INTENT(out), DIMENSION(:), ALLOCATABLE :: value !! Vector of output values
LOGICAL :: ret !! Return status (.true. on success)
INTEGER :: i,ns
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; ns = SIZE(str) ; ALLOCATE(value(ns))
DO i=1,ns
zs = remove_quotes(str(i))
IF (string_is(zs) /= st_logical) THEN
ret = .false. ; DEALLOCATE(value) ; RETURN
ELSE
READ(zs, *) value(i)
ENDIF
ENDDO
RETURN
END FUNCTION str2log_ve
FUNCTION str2real_ve(str, value) RESULT(ret)
!! Convert strings values to simple precision floating point values (vector)
CHARACTER(len=*), INTENT(in), DIMENSION(:) :: str !! Vector of strings to convert
REAL(kind=4), INTENT(out), DIMENSION(:), ALLOCATABLE :: value !! Vector of output values
LOGICAL :: ret !! Return status (.true. on success)
INTEGER :: i,ns
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; ns = SIZE(str) ; ALLOCATE(value(ns))
DO i=1,ns
IF (string_is(zs) < st_integer) THEN
ret = .false. ; DEALLOCATE(value) ; RETURN
ELSE
READ(zs, *) value(i)
ENDIF
ENDDO
RETURN
END FUNCTION str2real_ve
FUNCTION str2dble_ve(str, value) RESULT(ret)
!! Convert strings values to double precision floating point values (vector)
CHARACTER(len=*), INTENT(in), DIMENSION(:) :: str !! Vector of strings to convert
REAL(kind=8), INTENT(out), DIMENSION(:), ALLOCATABLE :: value !! Vector of output values
LOGICAL :: ret !! Return status (.true. on success)
INTEGER :: i,ns
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; ns = SIZE(str) ; ALLOCATE(value(ns))
DO i=1,ns
zs = remove_quotes(str(i))
IF (string_is(zs) < st_integer) THEN
ret = .false. ; DEALLOCATE(value) ; RETURN
ELSE
READ(zs, *) value(i)
ENDIF
ENDDO
RETURN
END FUNCTION str2dble_ve
FUNCTION str2cplx_ve(str, value) RESULT(ret)
!! Convert strings values to complex values (vector)
CHARACTER(len=*), INTENT(in), DIMENSION(:) :: str !! Vector of strings to convert
COMPLEX(kind=4), INTENT(out), DIMENSION(:), ALLOCATABLE :: value !! Vector of output values
LOGICAL :: ret !! Return status (.true. on success)
INTEGER :: i,ns
CHARACTER(len=:), ALLOCATABLE :: zs
ret = .true. ; ns = SIZE(str) ; ALLOCATE(value(ns))
DO i=1,ns
zs = remove_quotes(str(i))
IF (string_is(zs) /= st_complex) THEN
ret = .false. ; DEALLOCATE(value) ; RETURN
ELSE
READ(zs, *) value(i)
ENDIF
ENDDO
RETURN
END FUNCTION str2cplx_ve
FUNCTION int2str_as(value) RESULT(str)
!! Convert an integer value to string (auto format / string result)
INTEGER, INTENT(in) :: value !! Value to convert
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=DIGITS(value)) :: str)
WRITE(str,*,iostat=err) value
str = TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION int2str_as
FUNCTION log2str_as(value) RESULT(str)
!! Convert a logical value to string (auto format / string result)
LOGICAL, INTENT(in) :: value !! Value to convert
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=2) :: str)
WRITE(str, *, IOSTAT = err) value
str=TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION log2str_as
FUNCTION real2str_as(value) RESULT(str)
!! Convert a simple precision floating point value to string (auto format / string result)
REAL(kind=4), INTENT(in) :: value !! Value to convert
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=DIGITS(value)) ::str)
WRITE(str,*, IOSTAT = err) value
str=TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION real2str_as
FUNCTION dble2str_as(value) RESULT(str)
!! Convert a double precision floating point value to string (auto format / string result)
REAL(kind=8), INTENT(in) :: value !! Value to convert
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=DIGITS(value)) ::str)
WRITE(str,*, IOSTAT = err) value
str=TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION dble2str_as
FUNCTION cplx2str_as(value) RESULT(str)
!! Convert a complex value to string (auto format / string result)
COMPLEX(kind=4), INTENT(in) :: value !! Value to convert
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err,sl
sl = DIGITS(REAL(value))*2+3
ALLOCATE(CHARACTER(len=sl) :: str)
WRITE(str, *, IOSTAT = err) value
str = TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION cplx2str_as
FUNCTION int2str_fs(value, fmt, width) RESULT(str)
!! Convert an integer value to string (user format / string result)
INTEGER, INTENT(in) :: value !! Value to convert
CHARACTER(len=*), INTENT(in) :: fmt !! String format
INTEGER, INTENT(in) :: width !! Expected width of the output string (as defined in __fmt__)
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=width) :: str)
WRITE(str, '('//fmt//')', IOSTAT = err) value
str = TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION int2str_fs
FUNCTION log2str_fs(value, fmt, width) RESULT(str)
!! Convert a logical value to string (user format / string result)
LOGICAL, INTENT(in) :: value !! Value to convert
CHARACTER(len=*), INTENT(in) :: fmt !! String format
INTEGER, INTENT(in) :: width !! Expected width of the output string (as defined in __fmt__)
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=width) :: str)
WRITE(str, '('//fmt//')', IOSTAT = err) value
str=TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION log2str_fs
FUNCTION real2str_fs(value, fmt, width) RESULT(str)
!! Convert a simple precision floating point value to string (user format / string result)
REAL(kind=4), INTENT(in) :: value !! Value to convert
CHARACTER(len=*), INTENT(in) :: fmt !! String format
INTEGER, INTENT(in) :: width !! Expected width of the output string (as defined in __fmt__)
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=width) :: str)
WRITE(str, '('//fmt//')', IOSTAT = err) value
str = TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION real2str_fs
FUNCTION dble2str_fs(value, fmt, width) RESULT(str)
!! Convert a double precision floating point value to string (user format / string result)
REAL(kind=8), INTENT(in) :: value !! Value to convert
CHARACTER(len=*), INTENT(in) :: fmt !! String format
INTEGER, INTENT(in) :: width !! Expected width of the output string (as defined in __fmt__)
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=width) :: str)
WRITE(str, '('//fmt//')', IOSTAT = err) value
str = TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION dble2str_fs
FUNCTION cplx2str_fs(value, fmt, width) RESULT(str)
!! Convert a complex value to string (user format / string result)
COMPLEX(kind=4), INTENT(in) :: value !! Value to convert
CHARACTER(len=*), INTENT(in) :: fmt !! String format
INTEGER, INTENT(in) :: width !! Expected width of the output string (as defined in __fmt__)
CHARACTER(len=:), ALLOCATABLE :: str !! String with the converted value in output
INTEGER :: err
ALLOCATE(CHARACTER(len=width) :: str)
WRITE(str, '('//fmt//')', IOSTAT = err) value
str = TRIM(ADJUSTL(str))
IF (err /= 0) str = ''
RETURN
END FUNCTION cplx2str_fs
END MODULE STRINGS
MODULE ESTRINGS
!! Fortran strings extensions
!!
!! This module is an extension of [[strings(module)]] module (i.e. contains all its definitions).
!! It defines overloaded string concatenation (//) and assignment operators (=) that simplify the
!! conversion between intrinsic types and strings.
!!
!! These operators only work with allocatable strings.
USE STRINGS
PUBLIC
PRIVATE :: str_affect_int, str_affect_bool, str_affect_real, &
str_affect_double, str_affect_cplx, str_affect_dcplx, &
str_cat_int, str_cat_bool, str_cat_real, str_cat_double, &
str_cat_cplx, str_cat_dcplx, str_cat_int_inv, &
str_cat_bool_inv, str_cat_real_inv, str_cat_double_inv, &
str_cat_cplx_inv, str_cat_dcplx_inv
!> Overloaded string assignment operator interface
INTERFACE ASSIGNMENT(=)
MODULE PROCEDURE str_affect_int, str_affect_bool, str_affect_real, &
str_affect_double, str_affect_cplx, str_affect_dcplx
END INTERFACE
!> Overloaded string concatentation operator interface
INTERFACE OPERATOR(//)
MODULE PROCEDURE str_cat_int, str_cat_bool, str_cat_real, str_cat_double, &
str_cat_cplx, str_cat_dcplx
MODULE PROCEDURE str_cat_int_inv, str_cat_bool_inv, str_cat_real_inv, &
str_cat_double_inv, str_cat_cplx_inv, str_cat_dcplx_inv
END INTERFACE
CONTAINS
! Extended strings features
! ---------------------------
FUNCTION str_cat_int(str1,int2) RESULT(str)
!! Concatenate a string with a integer
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
INTEGER, INTENT(in) :: int2 !! Integer to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
ALLOCATE(CHARACTER(len=DIGITS(int2)) :: str)
WRITE(str,*) int2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str1//str
RETURN
END FUNCTION str_cat_int
!! @param[in] int2 An integer to concatenate
!! @param[in] str1 A string to concatenate
!! @return An allocatable string with the concatenation of input values.
FUNCTION str_cat_int_inv(int2,str1) RESULT(str)
!! Concatenate a string with a integer (inversed)
INTEGER, INTENT(in) :: int2 !! Integer to concatenate
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
ALLOCATE(CHARACTER(len=DIGITS(int2)) :: str)
WRITE(str,*) int2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_int_inv
FUNCTION str_cat_bool(str1,bool2) RESULT(str)
!! Concatenate a string with a logical
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
LOGICAL, INTENT(in) :: bool2 !! Logical to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
CHARACTER(len=2) :: tmp
WRITE(tmp,*) bool2
str=TRIM(ADJUSTL(tmp))
IF (LEN(str1) /= 0) str = str1//str
RETURN
END FUNCTION str_cat_bool
FUNCTION str_cat_bool_inv(bool2,str1) RESULT(str)
!! Concatenate a string with a logical (inversed)
LOGICAL, INTENT(in) :: bool2 !! Logical to concatenate
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
CHARACTER(len=2) :: tmp
WRITE(tmp,*) bool2
str = TRIM(ADJUSTL(tmp))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_bool_inv
FUNCTION str_cat_real(str1,real2) RESULT(str)
!! Concatenate a string with a real simple precision
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
REAL(kind=4), INTENT(in) :: real2 !! Simple precision real to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
ALLOCATE(CHARACTER(len=DIGITS(real2)) :: str)
WRITE(str,*) real2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str=str1//str
RETURN
END FUNCTION str_cat_real
FUNCTION str_cat_real_inv(real2,str1) RESULT(str)
!! Concatenate a string with a real simple precision (inversed)
REAL(kind=4), INTENT(in) :: real2 !! Simple precision real to concatenate
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
ALLOCATE(CHARACTER(len=DIGITS(real2)) :: str)
WRITE(str,*) real2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_real_inv
FUNCTION str_cat_double(str1,double2) RESULT(str)
!! Concatenate a string with a real double precision
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
REAL(kind=8), INTENT(in) :: double2 !! Double precision real to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
ALLOCATE(CHARACTER(len=DIGITS(double2)) :: str)
WRITE(str,*) double2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str=str1//str
RETURN
END FUNCTION str_cat_double
FUNCTION str_cat_double_inv(double2,str1) RESULT(str)
!! Concatenate a string with a real double precision (inversed)
REAL(kind=8), INTENT(in) :: double2 !! Double precision real to concatenate
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
ALLOCATE(CHARACTER(len=DIGITS(double2)) :: str)
WRITE(str,*) double2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_double_inv
FUNCTION str_cat_cplx(str1,cplx2) RESULT(str)
!! Concatenate a string with a complex
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
COMPLEX(kind=4), INTENT(in) :: cplx2 !! Complex value to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
INTEGER :: sl
sl = DIGITS(REAL(cplx2))*2+3
ALLOCATE(CHARACTER(len=sl) :: str)
WRITE(str,*) cplx2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_cplx
FUNCTION str_cat_cplx_inv(cplx2,str1) RESULT(str)
!! Concatenate a string with a complex (inversed)
COMPLEX(kind=4), INTENT(in) :: cplx2 !! Complex value to concatenate
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
INTEGER :: sl
sl = DIGITS(REAL(cplx2))*2+3
ALLOCATE(CHARACTER(len=sl) :: str)
WRITE(str,*) cplx2
str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_cplx_inv
FUNCTION str_cat_dcplx(str1,dcplx2) RESULT(str)
!! Concatenate a string with a double precision complex
CHARACTER(len=*), INTENT(in) :: str1 !! String to concatenate
COMPLEX(kind=8), INTENT(in) :: dcplx2 !! Complex value to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
INTEGER :: sl
sl = DIGITS(REAL(dcplx2))*2+3
ALLOCATE(CHARACTER(len=sl) :: str)
WRITE(str,*) dcplx2 ; str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_dcplx
FUNCTION str_cat_dcplx_inv(dcplx2,str1) RESULT(str)
!! Concatenate a string with a double precision complex (inversed)
COMPLEX(kind=8), INTENT(in) :: dcplx2 !! Complex value to concatenate
CHARACTER(len=*), INTENT(in) :: str1 !! string to concatenate
CHARACTER(len=:), ALLOCATABLE :: str !! Concatenation resulting string
INTEGER :: sl
sl = DIGITS(REAL(dcplx2))*2+3
ALLOCATE(CHARACTER(len=sl) :: str)
WRITE(str,*) dcplx2
str = TRIM(ADJUSTL(str))
IF (LEN(str1) /= 0) str = str//str1
RETURN
END FUNCTION str_cat_dcplx_inv
SUBROUTINE str_affect_int(str,int)
!! Assignment subroutine (using intrinsic integer)
CHARACTER(len=:), INTENT(out), ALLOCATABLE :: str !! Output string to be assigned
INTEGER, INTENT(in) :: int !! Input value to assign
str = str_cat_int('',int)
END SUBROUTINE str_affect_int
SUBROUTINE str_affect_bool(str,bool)
!! Assignment subroutine (using intrinsic logical)
CHARACTER(len=:), INTENT(out), ALLOCATABLE :: str !! Output string to be assigned
LOGICAL, INTENT(in) :: bool !! Input value to assign
str = str_cat_bool('',bool)
END SUBROUTINE str_affect_bool
SUBROUTINE str_affect_real(str,float)
!! Assignment subroutine (using intrinsic real)
CHARACTER(len=:), INTENT(out), ALLOCATABLE :: str !! Output string to be assigned
REAL(kind=4), INTENT(in) :: float !! Input value to assign
str = str_cat_real('',float)
END SUBROUTINE str_affect_real
SUBROUTINE str_affect_double(str,double)
!! Assignment subroutine (using intrinsic real(kind=8))
CHARACTER(len=:), INTENT(out), ALLOCATABLE :: str !! Output string to be assigned
REAL(kind=8), INTENT(in) :: double !! Input value to assign
str = str_cat_double('',double)
END SUBROUTINE str_affect_double
SUBROUTINE str_affect_cplx(str,cplx)
!! Assignment subroutine (using intrinsic complex)
CHARACTER(len=:), INTENT(out), ALLOCATABLE :: str !! Output string to be assigned
COMPLEX(kind=4), INTENT(in) :: cplx !! Input value to assign
str = str_cat_cplx('',cplx)
END SUBROUTINE str_affect_cplx
SUBROUTINE str_affect_dcplx(str,dcplx)
!! Assignment subroutine (using intrinsic complex(kind=8))
CHARACTER(len=:), INTENT(out), ALLOCATABLE :: str !! Output string to be assigned
COMPLEX(kind=8), INTENT(in) :: dcplx !! Input value to assign
str = str_cat_dcplx('',dcplx)
END SUBROUTINE str_affect_dcplx
END MODULE estrings