/*
Facebook plugin for Miranda Instant Messenger
_____________________________________________
Copyright © 2009-11 Michal Zelinka, 2011-12 Robert Pösel
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#include "common.h"
std::string utils::url::encode(const std::string &s)
{
char *encoded = reinterpret_cast(CallService( MS_NETLIB_URLENCODE,
0,reinterpret_cast(s.c_str())));
std::string ret = encoded;
HeapFree(GetProcessHeap(),0,encoded);
return ret;
}
std::string utils::url::decode(std::string data)
{
// TODO: Better and universal method?
utils::text::replace_all( &data, "%2F", "/" );
utils::text::replace_all( &data, "%3F", "?" );
utils::text::replace_all( &data, "%3D", "=" );
utils::text::replace_all( &data, "%26", "&" );
return data;
}
std::string utils::time::unix_timestamp( )
{
time_t in = ::time( NULL );
return utils::conversion::to_string( ( void* )&in, UTILS_CONV_TIME_T );
}
std::string utils::time::mili_timestamp( )
{
SYSTEMTIME st;
std::string timestamp = utils::time::unix_timestamp();
GetSystemTime(&st);
timestamp.append(utils::conversion::to_string( ( void* )&st.wMilliseconds, UTILS_CONV_UNSIGNED_NUMBER ));
return timestamp;
}
DWORD utils::time::fix_timestamp( double mili_timestamp )
{
// If it is really mili_timestamp
if (mili_timestamp > 100000000000) {
return (DWORD) (mili_timestamp / 1000);
}
return (DWORD) mili_timestamp;
}
DWORD utils::conversion::to_timestamp( std::string data )
{
DWORD timestamp = NULL;
if (!utils::conversion::from_string(timestamp, data, std::dec)) {
timestamp = static_cast(::time(NULL));
}
return timestamp;
}
std::string utils::conversion::to_string( void* data, WORD type )
{
std::stringstream out;
switch ( type )
{
case UTILS_CONV_BOOLEAN:
out << (data ? "true" : "false");
case UTILS_CONV_TIME_T:
out << (*( time_t* )data);
break;
case UTILS_CONV_SIGNED_NUMBER:
out << (*( signed int* )data);
break;
case UTILS_CONV_UNSIGNED_NUMBER:
out << (*( unsigned int* )data);
break;
}
return out.str( );
}
void utils::text::replace_first( std::string* data, std::string from, std::string to )
{
std::string::size_type position = data->find(from);
if ( position != std::string::npos )
{
data->replace( position, from.size(), to );
}
}
void utils::text::replace_all( std::string* data, std::string from, std::string to )
{
std::string::size_type position = 0;
while ( ( position = data->find( from, position )) != std::string::npos )
{
data->replace( position, from.size(), to );
position++;
}
}
unsigned int utils::text::count_all( std::string* data, std::string term )
{
unsigned int count = 0;
std::string::size_type position = 0;
while ( ( position = data->find( term, position )) != std::string::npos )
{
count++;
position++;
}
return count;
}
std::string utils::text::special_expressions_decode( std::string data )
{
utils::text::replace_all( &data, "&", "&" );
utils::text::replace_all( &data, """, "\"" );
utils::text::replace_all( &data, "'", "'" );
utils::text::replace_all( &data, "@", "@" );
utils::text::replace_all( &data, "<", "<" );
utils::text::replace_all( &data, ">", ">" );
utils::text::replace_all( &data, "♥", "\xE2\x99\xA5" ); // direct byte replacement
// utils::text::replace_all( &data, "♥", "\\u2665" ); // indirect slashu replacement
utils::text::replace_all( &data, "\\/", "/" );
utils::text::replace_all( &data, "\\\\", "\\" );
// TODO: Add more to comply general usage
// http://www.utexas.edu/learn/html/spchar.html
// http://www.webmonkey.com/reference/Special_Characters
// http://www.degraeve.com/reference/specialcharacters.php
// http://www.chami.com/tips/internet/050798i.html
// http://www.w3schools.com/tags/ref_entities.asp
// http://www.natural-innovations.com/wa/doc-charset.html
// http://webdesign.about.com/library/bl_htmlcodes.htm
return data;
}
std::string utils::text::edit_html( std::string data )
{
std::string::size_type end = 0;
std::string::size_type start = 0;
std::string new_string = "";
while ( end != std::string::npos )
{
end = data.find( "", start );
if ( end != std::string::npos )
{
new_string += data.substr( start, end - start );
start = data.find( "<\\/div", end );
} else {
new_string += data.substr( start, data.length() - start );
}
}
// Append newline after attachement title
start = new_string.find( "class=\\\"uiAttachmentTitle", 0 );
if ( start != std::string::npos )
{
data = new_string.substr( 0, start );
data = utils::text::trim( data );
start = new_string.find( ">", start );
if ( start != std::string::npos )
new_string.insert(start+1, "\n\n");
start = new_string.find( "<\\/div>", start );
if ( start != std::string::npos )
new_string.insert(start, "\n");
}
// Append newline between attachement link and description
start = new_string.find( "uiAttachmentDesc", 0 );
if ( start != std::string::npos )
{
start = new_string.find( ">", start );
if ( start != std::string::npos )
new_string.insert(start+1, "\n");
start = new_string.find( "<\\/div>", start );
if ( start != std::string::npos )
new_string.insert(start, "\n");
}
utils::text::replace_all( &new_string, "
", "\n" );
utils::text::replace_all( &new_string, "\n\n\n", "\n\n" );
//utils::text::replace_all( &new_string, "\\t", "" );
//utils::text::replace_all( &new_string, "\\n", "" );
return new_string;
}
std::string utils::text::remove_html( std::string data )
{
std::string new_string = "";
for ( std::string::size_type i = 0; i < data.length( ); i++ )
{
if ( data.at(i) == '<' && data.at(i+1) != ' ' )
{
i = data.find( ">", i );
if (i == std::string::npos)
break;
continue;
}
new_string += data.at(i);
}
return new_string;
}
std::string utils::text::slashu_to_utf8( std::string data )
{
std::string new_string = "";
for ( std::string::size_type i = 0; i < data.length( ); i++ )
{
if ( data.at(i) == '\\' && (i+1) < data.length( ) && data.at(i+1) == 'u' )
{
unsigned int udn = strtol( data.substr( i + 2, 4 ).c_str(), NULL, 16 );
if ( udn >= 128 && udn <= 2047 )
{ // U+0080 .. U+07FF
new_string += ( char )( 192 + ( udn / 64 ));
new_string += ( char )( 128 + ( udn % 64 ));
}
else if ( udn >= 2048 && udn <= 65535 )
{ // U+0800 .. U+FFFF
new_string += ( char )( 224 + ( udn / 4096 ));
new_string += ( char )( 128 + ( ( udn / 64 ) % 64 ));
new_string += ( char )( 128 + ( udn % 64 ));
}
else if ( udn <= 127 )
{ // U+0000 .. U+007F (should not appear)
new_string += ( char )udn;
}
i += 5;
continue;
}
new_string += data.at(i);
}
return new_string;
}
std::string utils::text::trim( std::string data )
{
std::string spaces = " \t\r\n"; // TODO: include "nbsp"?
std::string::size_type begin = data.find_first_not_of( spaces );
std::string::size_type end = data.find_last_not_of( spaces ) + 1;
return (begin != std::string::npos) ? data.substr( begin, end - begin ) : "";
}
void utils::text::explode(std::string str, std::string separator, std::vector* results)
{
std::string::size_type pos;
pos = str.find_first_of(separator);
while (pos != std::string::npos) {
if (pos > 0) {
results->push_back(str.substr(0,pos));
}
str = str.substr(pos+1);
pos = str.find_first_of(separator);
}
if (str.length() > 0) {
results->push_back(str);
}
}
std::string utils::text::source_get_value( std::string* data, unsigned int argument_count, ... )
{
va_list arg;
std::string ret;
std::string::size_type start = 0, end = 0;
va_start( arg, argument_count );
for ( unsigned int i = argument_count; i > 0; i-- )
{
if ( i == 1 )
{
end = data->find( va_arg( arg, char* ), start );
if ( start == std::string::npos || end == std::string::npos )
break;
ret = data->substr( start, end - start );
} else {
std::string term = va_arg( arg, char* );
start = data->find( term, start );
if ( start == std::string::npos )
break;
start += term.length();
}
}
va_end( arg );
return ret;
}
std::string utils::text::source_get_value2( std::string* data, const char *term, const char *endings)
{
std::string::size_type start = 0, end = 0;
std::string ret;
start = data->find(term);
if (start != std::string::npos) {
start += strlen(term);
end = data->find_first_of(endings, start);
if (end != std::string::npos) {
ret = data->substr( start, end - start );
}
}
return ret;
}
int utils::number::random( )
{
srand( ::time( NULL ));
return rand( );
}
int utils::debug::log(std::string file_name, std::string text)
{
char szFile[MAX_PATH];
GetModuleFileNameA(g_hInstance, szFile, SIZEOF(szFile));
std::string path = szFile;
path = path.substr( 0, path.rfind( "\\" ));
path = path.substr( 0, path.rfind( "\\" ) + 1 );
path = path + file_name.c_str() + ".txt";
SYSTEMTIME time;
GetLocalTime( &time );
std::ofstream out( path.c_str(), std::ios_base::out | std::ios_base::app | std::ios_base::ate );
out << "[" << (time.wHour < 10 ? "0" : "") << time.wHour << ":" << (time.wMinute < 10 ? "0" : "") << time.wMinute << ":" << (time.wSecond < 10 ? "0" : "") << time.wSecond << "] " << text << std::endl;
out.close( );
return EXIT_SUCCESS;
}
void __fastcall utils::mem::detract(char** str )
{
utils::mem::detract( ( void** )str );
}
void __fastcall utils::mem::detract(void** p)
{
utils::mem::detract((void*)(*p));
}
void __fastcall utils::mem::detract(void* p)
{
mir_free(p);
}
void* __fastcall utils::mem::allocate(size_t size)
{
return mir_calloc(size);
}
struct
{
char *ext;
int fmt;
}
static formats[] = {
{ ".png", PA_FORMAT_PNG },
{ ".jpg", PA_FORMAT_JPEG },
{ ".jpeg", PA_FORMAT_JPEG },
{ ".ico", PA_FORMAT_ICON },
{ ".bmp", PA_FORMAT_BMP },
{ ".gif", PA_FORMAT_GIF },
};
int ext_to_format(const std::string &ext)
{
for(size_t i=0; i