funcs = array_keys( $makepot->rules ); } public function usage() { $usage = "Usage: php add-textdomain.php [-i] \n\nAdds the string as a last argument to all i18n function calls in \nand prints the modified php file on standard output.\n\nOptions:\n -i Modifies the PHP file in place, instead of printing it to standard output.\n"; fwrite( STDERR, $usage ); exit( 1 ); } /** * Add textdomain to a single file. * * @see AddTextdomain::process_string() * * @param string $domain Text domain. * @param string $source_filename Filename with optional path. * @param bool $inplace True to modifies the PHP file in place. False to print to standard output. */ public function process_file( $domain, $source_filename, $inplace ) { $new_source = $this->process_string( $domain, file_get_contents( $source_filename ) ); if ($inplace) { $f = fopen( $source_filename, 'w' ); fwrite( $f, $new_source ); fclose( $f ); } else { echo $new_source; } } /** * Add textdomain to a string of PHP. * * Functions calls should be wrapped in opening and closing PHP delimiters as usual. * * @see AddTextdomain::process_tokens() * * @param string $domain Text domain. * @param string $string PHP code to parse. * * @return string Modified source. */ public function process_string( $domain, $string ) { $tokens = token_get_all( $string ); return $this->process_tokens( $domain, $tokens ); } /** * Add textdomain to a set of PHP tokens. * * @param string $domain Text domain. * @param array $tokens PHP tokens. An array of token identifiers. Each individual token identifier is either a * single character (i.e.: ;, ., >, !, etc.), or a three element array containing the token * index in element 0, the string content of the original token in element 1 and the line * number in element 2. * * @return string Modified source. */ public function process_tokens( $domain, $tokens ) { $this->modified_contents = ''; $domain = addslashes( $domain ); $in_func = false; $args_started = false; $parens_balance = 0; $found_domain = false; foreach( $tokens as $index => $token ) { $string_success = false; if ( is_array( $token ) ) { list( $id, $text ) = $token; if ( T_STRING == $id && in_array($text, $this->funcs ) ) { $in_func = true; $parens_balance = 0; $args_started = false; $found_domain = false; } elseif ( T_CONSTANT_ENCAPSED_STRING == $id && ( "'$domain'" == $text || "\"$domain\"" == $text ) ) { if ( $in_func && $args_started ) { $found_domain = true; } } $token = $text; } elseif ( '(' == $token ) { $args_started = true; ++$parens_balance; } elseif ( ')' == $token ) { --$parens_balance; if ( $in_func && 0 == $parens_balance ) { if ( ! $found_domain ) { $token = ", '$domain'"; if ( T_WHITESPACE == $tokens[ $index - 1 ][0] ) { $token .= ' '; // Maintain code standards if previously present // Remove previous whitespace token to account for it. $this->modified_contents = trim( $this->modified_contents ); } $token .= ')'; } $in_func = false; $args_started = false; $found_domain = false; } } $this->modified_contents .= $token; } return $this->modified_contents; } } // Run the CLI only if the file wasn't included. $included_files = get_included_files(); if ( __FILE__ == $included_files[0] ) { $adddomain = new AddTextdomain; if ( ! isset( $argv[1] ) || ! isset( $argv[2] ) ) { $adddomain->usage(); } $inplace = false; if ( '-i' == $argv[1] ) { $inplace = true; if ( !isset( $argv[3] ) ) { $adddomain->usage(); } array_shift( $argv ); } $adddomain->process_file( $argv[1], $argv[2], $inplace ); }