forked from KolibriOS/kolibrios
WebView 1.61: fix of downloading timeout, ultimate speedup; thanks hidnplayr for idea use event EVN_STACK instead of timeout
git-svn-id: svn://kolibrios.org@6978 a494cfbc-eb01-0410-851d-a64ba20cac60
This commit is contained in:
parent
1c6bb84e7f
commit
ad6c634152
@ -1,6 +1,6 @@
|
|||||||
//HTML Viewer in C--
|
//HTML Viewer in C--
|
||||||
//Copyright 2007-2013 by Veliant & Leency
|
//Copyright 2007-2017 by Veliant & Leency
|
||||||
//Asper, lev, Lrz, Barsuk, Nable...
|
//Asper, lev, Lrz, Barsuk, Nable, hidnplayr...
|
||||||
|
|
||||||
#ifndef AUTOBUILD
|
#ifndef AUTOBUILD
|
||||||
#include "lang.h--"
|
#include "lang.h--"
|
||||||
@ -30,7 +30,7 @@
|
|||||||
char homepage[] = FROM "html\\homepage.htm""\0";
|
char homepage[] = FROM "html\\homepage.htm""\0";
|
||||||
|
|
||||||
#ifdef LANG_RUS
|
#ifdef LANG_RUS
|
||||||
char version[]="’¥ªáâ®¢ë© ¡à 㧥à 1.6";
|
char version[]="’¥ªáâ®¢ë© ¡à 㧥à 1.61";
|
||||||
?define IMAGES_CACHE_CLEARED "Šíè ª à⨮ª ®ç¨é¥"
|
?define IMAGES_CACHE_CLEARED "Šíè ª à⨮ª ®ç¨é¥"
|
||||||
?define T_LAST_SLIDE "<EFBFBD>â® ¯®á«¥¤¨© á« ©¤"
|
?define T_LAST_SLIDE "<EFBFBD>â® ¯®á«¥¤¨© á« ©¤"
|
||||||
char loading[] = "‡ £à㧪 áâà ¨æë...<br>";
|
char loading[] = "‡ £à㧪 áâà ¨æë...<br>";
|
||||||
@ -43,7 +43,7 @@ char rmb_menu[] =
|
|||||||
Žç¨áâ¨âì ªíè ª à⨮ª
|
Žç¨áâ¨âì ªíè ª à⨮ª
|
||||||
Œ¥¥¤¦¥à § £à㧮ª";
|
Œ¥¥¤¦¥à § £à㧮ª";
|
||||||
#else
|
#else
|
||||||
char version[]="Text-based Browser 1.6";
|
char version[]="Text-based Browser 1.61";
|
||||||
?define IMAGES_CACHE_CLEARED "Images cache cleared"
|
?define IMAGES_CACHE_CLEARED "Images cache cleared"
|
||||||
?define T_LAST_SLIDE "This slide is the last"
|
?define T_LAST_SLIDE "This slide is the last"
|
||||||
char loading[] = "Loading...<br>";
|
char loading[] = "Loading...<br>";
|
||||||
@ -126,140 +126,137 @@ void main()
|
|||||||
if (param) strcpy(#URL, #param); else strcpy(#URL, URL_SERVICE_HOME);
|
if (param) strcpy(#URL, #param); else strcpy(#URL, URL_SERVICE_HOME);
|
||||||
WB1.list.SetFont(8, 14, 10011000b);
|
WB1.list.SetFont(8, 14, 10011000b);
|
||||||
WB1.list.no_selection = true;
|
WB1.list.no_selection = true;
|
||||||
SetEventMask(0xa7);
|
SetEventMask(EVM_REDRAW + EVM_KEY + EVM_BUTTON + EVM_MOUSE + EVM_MOUSE_FILTER + EVM_STACK);
|
||||||
BEGIN_LOOP_APPLICATION:
|
loop() switch(WaitEvent())
|
||||||
WaitEventTimeout(2);
|
{
|
||||||
switch(EAX & 0xFF)
|
case evMouse:
|
||||||
{
|
if (!CheckActiveProcess(Form.ID)) break;
|
||||||
CASE evMouse:
|
edit_box_mouse stdcall (#address_box);
|
||||||
if (!CheckActiveProcess(Form.ID)) break;
|
mouse.get();
|
||||||
edit_box_mouse stdcall (#address_box);
|
if (WB1.list.MouseOver(mouse.x, mouse.y))
|
||||||
mouse.get();
|
{
|
||||||
if (WB1.list.MouseOver(mouse.x, mouse.y))
|
if (PageLinks.HoverAndProceed(mouse.x, WB1.list.first + mouse.y))
|
||||||
{
|
&& (bufsize) && (mouse.pkm) && (mouse.up) {
|
||||||
if (PageLinks.HoverAndProceed(mouse.x, WB1.list.first + mouse.y))
|
EventShowPageMenu(mouse.x, mouse.y);
|
||||||
&& (bufsize) && (mouse.pkm) && (mouse.up) {
|
|
||||||
EventShowPageMenu(mouse.x, mouse.y);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (WB1.list.MouseScroll(mouse.vert)) WB1.DrawPage();
|
|
||||||
}
|
|
||||||
scrollbar_v_mouse (#scroll_wv);
|
|
||||||
if (WB1.list.first != scroll_wv.position)
|
|
||||||
{
|
|
||||||
WB1.list.first = scroll_wv.position;
|
|
||||||
WB1.DrawPage();
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
if (WB1.list.MouseScroll(mouse.vert)) WB1.DrawPage();
|
||||||
|
}
|
||||||
|
scrollbar_v_mouse (#scroll_wv);
|
||||||
|
if (WB1.list.first != scroll_wv.position)
|
||||||
|
{
|
||||||
|
WB1.list.first = scroll_wv.position;
|
||||||
|
WB1.DrawPage();
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case evButton:
|
case evButton:
|
||||||
ProcessEvent(GetButtonID());
|
ProcessEvent(GetButtonID());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case evKey:
|
case evKey:
|
||||||
GetKeys();
|
GetKeys();
|
||||||
if (address_box.flags & 0b10)
|
if (address_box.flags & 0b10)
|
||||||
|
{
|
||||||
|
if (key_ascii == ASCII_KEY_ENTER) ProcessEvent(key_scancode); else {
|
||||||
|
EAX = key_editbox;
|
||||||
|
edit_box_key stdcall(#address_box);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (WB1.list.ProcessKey(key_scancode)) WB1.DrawPage();
|
||||||
|
else ProcessEvent(key_scancode);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case evReDraw:
|
||||||
|
if (menu.list.cur_y) {
|
||||||
|
ProcessEvent(menu.list.cur_y);
|
||||||
|
menu.list.cur_y = 0;
|
||||||
|
}
|
||||||
|
DefineAndDrawWindow(GetScreenWidth()-800/2-random(80),GetScreenHeight()-600/2-random(80),800,600,0x73,col_bg,0,0);
|
||||||
|
GetProcessInfo(#Form, SelfInfo);
|
||||||
|
if (Form.status_window>2) { DrawTitle(#header); break; }
|
||||||
|
if (Form.height<120) { MoveSize(OLD,OLD,OLD,120); break; }
|
||||||
|
if (Form.width<280) { MoveSize(OLD,OLD,280,OLD); break; }
|
||||||
|
Draw_Window();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case evNetwork:
|
||||||
|
if (http_transfer > 0) {
|
||||||
|
http_receive stdcall (http_transfer);
|
||||||
|
$push EAX
|
||||||
|
ESI = http_transfer;
|
||||||
|
wv_progress_bar.max = ESI.http_msg.content_length;
|
||||||
|
if (wv_progress_bar.value != ESI.http_msg.content_received)
|
||||||
{
|
{
|
||||||
if (key_ascii == ASCII_KEY_ENTER) ProcessEvent(key_scancode); else {
|
wv_progress_bar.value = ESI.http_msg.content_received;
|
||||||
EAX = key_editbox;
|
DrawProgress();
|
||||||
edit_box_key stdcall(#address_box);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
$pop EAX
|
||||||
{
|
if (EAX == 0) {
|
||||||
if (WB1.list.ProcessKey(key_scancode)) WB1.DrawPage();
|
|
||||||
else ProcessEvent(key_scancode);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case evReDraw:
|
|
||||||
if (menu.list.cur_y) {
|
|
||||||
ProcessEvent(menu.list.cur_y);
|
|
||||||
menu.list.cur_y = 0;
|
|
||||||
}
|
|
||||||
DefineAndDrawWindow(GetScreenWidth()-800/2-random(80),GetScreenHeight()-600/2-random(80),800,600,0x73,col_bg,0,0);
|
|
||||||
GetProcessInfo(#Form, SelfInfo);
|
|
||||||
if (Form.status_window>2) { DrawTitle(#header); break; }
|
|
||||||
if (Form.height<120) { MoveSize(OLD,OLD,OLD,120); break; }
|
|
||||||
if (Form.width<280) { MoveSize(OLD,OLD,280,OLD); break; }
|
|
||||||
Draw_Window();
|
|
||||||
break;
|
|
||||||
|
|
||||||
case evNetwork:
|
|
||||||
if (http_transfer > 0) {
|
|
||||||
http_receive stdcall (http_transfer);
|
|
||||||
$push EAX
|
|
||||||
ESI = http_transfer;
|
ESI = http_transfer;
|
||||||
wv_progress_bar.max = ESI.http_msg.content_length;
|
// Handle redirects
|
||||||
if (wv_progress_bar.value != ESI.http_msg.content_received)
|
if (ESI.http_msg.status >= 300) && (ESI.http_msg.status < 400)
|
||||||
{
|
{
|
||||||
wv_progress_bar.value = ESI.http_msg.content_received;
|
redirected++;
|
||||||
DrawProgress();
|
if (redirected<=5)
|
||||||
}
|
|
||||||
$pop EAX
|
|
||||||
if (EAX == 0) {
|
|
||||||
ESI = http_transfer;
|
|
||||||
// Handle redirects
|
|
||||||
if (ESI.http_msg.status >= 300) && (ESI.http_msg.status < 400)
|
|
||||||
{
|
{
|
||||||
redirected++;
|
http_find_header_field stdcall (http_transfer, "location\0");
|
||||||
if (redirected<=5)
|
if (EAX!=0) {
|
||||||
{
|
ESI = EAX;
|
||||||
http_find_header_field stdcall (http_transfer, "location\0");
|
EDI = #URL;
|
||||||
if (EAX!=0) {
|
do {
|
||||||
ESI = EAX;
|
$lodsb;
|
||||||
EDI = #URL;
|
$stosb;
|
||||||
do {
|
} while (AL != 0) && (AL != 13) && (AL != 10);
|
||||||
$lodsb;
|
DSBYTE[EDI-1]='\0';
|
||||||
$stosb;
|
if (!strncmp(#URL,"https://",8))
|
||||||
} while (AL != 0) && (AL != 13) && (AL != 10);
|
{
|
||||||
DSBYTE[EDI-1]='\0';
|
ShowErrorMessageThatHttpsIsNotSupportedYet();
|
||||||
if (!strncmp(#URL,"https://",8))
|
StopLoading();
|
||||||
{
|
break;
|
||||||
ShowErrorMessageThatHttpsIsNotSupportedYet();
|
|
||||||
StopLoading();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
notify("Too many redirects");
|
|
||||||
StopLoading();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
redirected = 0;
|
|
||||||
}
|
|
||||||
// Loading the page is complete, free resources
|
|
||||||
if (redirected>0)
|
|
||||||
{
|
|
||||||
http_free stdcall (http_transfer);
|
|
||||||
http_transfer=0;
|
|
||||||
GetAbsoluteURL(#URL);
|
|
||||||
history.back();
|
|
||||||
strcpy(#editURL, #URL);
|
|
||||||
DrawEditBoxWebView();
|
|
||||||
OpenPage();
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
history.add(#URL);
|
notify("Too many redirects");
|
||||||
ESI = http_transfer;
|
StopLoading();
|
||||||
bufpointer = ESI.http_msg.content_ptr;
|
break;
|
||||||
bufsize = ESI.http_msg.content_received;
|
|
||||||
http_free stdcall (http_transfer);
|
|
||||||
http_transfer=0;
|
|
||||||
SetPageDefaults();
|
|
||||||
ShowPage();
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
redirected = 0;
|
||||||
|
}
|
||||||
|
// Loading the page is complete, free resources
|
||||||
|
if (redirected>0)
|
||||||
|
{
|
||||||
|
http_free stdcall (http_transfer);
|
||||||
|
http_transfer=0;
|
||||||
|
GetAbsoluteURL(#URL);
|
||||||
|
history.back();
|
||||||
|
strcpy(#editURL, #URL);
|
||||||
|
DrawEditBoxWebView();
|
||||||
|
OpenPage();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
history.add(#URL);
|
||||||
|
ESI = http_transfer;
|
||||||
|
bufpointer = ESI.http_msg.content_ptr;
|
||||||
|
bufsize = ESI.http_msg.content_received;
|
||||||
|
http_free stdcall (http_transfer);
|
||||||
|
http_transfer=0;
|
||||||
|
SetPageDefaults();
|
||||||
|
ShowPage();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
goto BEGIN_LOOP_APPLICATION;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetElementSizes()
|
void SetElementSizes()
|
||||||
|
@ -39,7 +39,7 @@ void Downloader()
|
|||||||
int key, i;
|
int key, i;
|
||||||
char notify_message[4296];
|
char notify_message[4296];
|
||||||
downloader_opened = 1;
|
downloader_opened = 1;
|
||||||
SetEventMask(0x27);
|
SetEventMask(EVM_REDRAW + EVM_KEY + EVM_BUTTON + EVM_MOUSE + EVM_MOUSE_FILTER + EVM_STACK);
|
||||||
|
|
||||||
system.color.get();
|
system.color.get();
|
||||||
pb.frame_color = system.color.work_dark;
|
pb.frame_color = system.color.work_dark;
|
||||||
@ -50,69 +50,65 @@ void Downloader()
|
|||||||
if (downloader_edit[0]) StartDownloading(); else strcpy(#downloader_edit, "http://");
|
if (downloader_edit[0]) StartDownloading(); else strcpy(#downloader_edit, "http://");
|
||||||
ed.size = ed.pos = ed.shift = ed.shift_old = strlen(#downloader_edit);
|
ed.size = ed.pos = ed.shift = ed.shift_old = strlen(#downloader_edit);
|
||||||
|
|
||||||
loop()
|
loop() switch(WaitEvent())
|
||||||
{
|
{
|
||||||
WaitEventTimeout(30);
|
case evMouse:
|
||||||
switch(EAX & 0xFF)
|
if (!CheckActiveProcess(DL_Form.ID)) break;
|
||||||
{
|
edit_box_mouse stdcall (#ed);
|
||||||
CASE evMouse:
|
break;
|
||||||
if (!CheckActiveProcess(DL_Form.ID)) break;
|
|
||||||
edit_box_mouse stdcall (#ed);
|
case evButton:
|
||||||
break;
|
Key_Scan(GetButtonID());
|
||||||
|
break;
|
||||||
|
|
||||||
|
case evKey:
|
||||||
|
GetKeys();
|
||||||
|
edit_box_key stdcall(#ed);
|
||||||
|
if (key_scancode==SCAN_CODE_ENTER) Key_Scan(301);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case evReDraw:
|
||||||
|
DefineAndDrawWindow(215, 100, 580, 130, 0x74, system.color.work, DL_WINDOW_HEADER, 0);
|
||||||
|
GetProcessInfo(#DL_Form, SelfInfo);
|
||||||
|
if (DL_Form.status_window>2) break;
|
||||||
|
if (DL_Form.height<120) MoveSize(OLD,OLD,OLD,120);
|
||||||
|
if (DL_Form.width<280) MoveSize(OLD,OLD,280,OLD);
|
||||||
|
DL_Draw_Window();
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (!downloader.MonitorProgress()) break;
|
||||||
|
pb.max = downloader.data_full_size;
|
||||||
|
if (pb.value != downloader.data_downloaded_size)
|
||||||
|
{
|
||||||
|
pb.value = downloader.data_downloaded_size;
|
||||||
|
progressbar_draw stdcall(#pb);
|
||||||
|
DrawDownloading();
|
||||||
|
}
|
||||||
|
if (downloader.state == STATE_COMPLETED)
|
||||||
|
{
|
||||||
|
if (!dir_exists(#save_to)) CreateDir(#save_to);
|
||||||
|
strcpy(#filepath, #save_to);
|
||||||
|
chrcat(#filepath, '/');
|
||||||
|
// Clean all slashes at the end
|
||||||
|
strcpy(#aux, #downloader_edit);
|
||||||
|
while (aux[strlen(#aux)-1] == '/') {
|
||||||
|
aux[strlen(#aux)-1] = 0;
|
||||||
|
}
|
||||||
|
strcat(#filepath, #aux+strrchr(#aux, '/'));
|
||||||
|
|
||||||
|
for (i=0; i<strlen(#filepath); i++) if(filepath[i]==':')||(filepath[i]=='?')filepath[i]='-';
|
||||||
|
|
||||||
case evButton:
|
if (WriteFile(downloader.data_downloaded_size, downloader.bufpointer, #filepath)==0)
|
||||||
Key_Scan(GetButtonID());
|
sprintf(#notify_message, "%s%s%s",FILE_SAVED_AS,#filepath,"' -Dt");
|
||||||
break;
|
else
|
||||||
|
sprintf(#notify_message, "%s%s%s","'Download manager\nError! Can\96t save file as ",#filepath,"' -Et");
|
||||||
case evKey:
|
|
||||||
GetKeys();
|
notify(#notify_message);
|
||||||
edit_box_key stdcall(#ed);
|
StopDownloading();
|
||||||
if (key_scancode==SCAN_CODE_ENTER) Key_Scan(301);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case evReDraw:
|
|
||||||
DefineAndDrawWindow(215, 100, 580, 130, 0x74, system.color.work, DL_WINDOW_HEADER, 0);
|
|
||||||
GetProcessInfo(#DL_Form, SelfInfo);
|
|
||||||
if (DL_Form.status_window>2) break;
|
|
||||||
if (DL_Form.height<120) MoveSize(OLD,OLD,OLD,120);
|
|
||||||
if (DL_Form.width<280) MoveSize(OLD,OLD,280,OLD);
|
|
||||||
DL_Draw_Window();
|
DL_Draw_Window();
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
default:
|
|
||||||
if (!downloader.MonitorProgress()) break;
|
|
||||||
pb.max = downloader.data_full_size;
|
|
||||||
if (pb.value != downloader.data_downloaded_size)
|
|
||||||
{
|
|
||||||
pb.value = downloader.data_downloaded_size;
|
|
||||||
progressbar_draw stdcall(#pb);
|
|
||||||
DrawDownloading();
|
|
||||||
}
|
|
||||||
if (downloader.state == STATE_COMPLETED)
|
|
||||||
{
|
|
||||||
if (!dir_exists(#save_to)) CreateDir(#save_to);
|
|
||||||
strcpy(#filepath, #save_to);
|
|
||||||
chrcat(#filepath, '/');
|
|
||||||
// Clean all slashes at the end
|
|
||||||
strcpy(#aux, #downloader_edit);
|
|
||||||
while (aux[strlen(#aux)-1] == '/') {
|
|
||||||
aux[strlen(#aux)-1] = 0;
|
|
||||||
}
|
|
||||||
strcat(#filepath, #aux+strrchr(#aux, '/'));
|
|
||||||
|
|
||||||
for (i=0; i<strlen(#filepath); i++) if(filepath[i]==':')||(filepath[i]=='?')filepath[i]='-';
|
|
||||||
|
|
||||||
if (WriteFile(downloader.data_downloaded_size, downloader.bufpointer, #filepath)==0)
|
|
||||||
sprintf(#notify_message, "%s%s%s",FILE_SAVED_AS,#filepath,"' -Dt");
|
|
||||||
else
|
|
||||||
sprintf(#notify_message, "%s%s%s","'Download manager\nError! Can\96t save file as ",#filepath,"' -Et");
|
|
||||||
|
|
||||||
notify(#notify_message);
|
|
||||||
StopDownloading();
|
|
||||||
DL_Draw_Window();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,12 +36,28 @@ char program_path[4096];
|
|||||||
#define evReDraw 1
|
#define evReDraw 1
|
||||||
#define evKey 2
|
#define evKey 2
|
||||||
#define evButton 3
|
#define evButton 3
|
||||||
|
#define evExit 4
|
||||||
#define evDesktop 5
|
#define evDesktop 5
|
||||||
#define evMouse 6
|
#define evMouse 6
|
||||||
#define evIPC 7
|
#define evIPC 7
|
||||||
#define evNetwork 8
|
#define evNetwork 8
|
||||||
#define evDebug 9
|
#define evDebug 9
|
||||||
|
|
||||||
|
//Event mask bits for function 40
|
||||||
|
#define EVM_REDRAW 1b
|
||||||
|
#define EVM_KEY 10b
|
||||||
|
#define EVM_BUTTON 100b
|
||||||
|
#define EVM_EXIT 1000b
|
||||||
|
#define EVM_BACKGROUND 10000b
|
||||||
|
#define EVM_MOUSE 100000b
|
||||||
|
#define EVM_IPC 1000000b
|
||||||
|
#define EVM_STACK 10000000b
|
||||||
|
#define EVM_DEBUG 100000000b
|
||||||
|
#define EVM_STACK2 1000000000b
|
||||||
|
#define EVM_MOUSE_FILTER 0x80000000
|
||||||
|
#define EVM_CURSOR_FILTER 0x40000000
|
||||||
|
|
||||||
|
|
||||||
//Button options
|
//Button options
|
||||||
#define BT_DEL 0x80000000
|
#define BT_DEL 0x80000000
|
||||||
#define BT_HIDE 0x40000000
|
#define BT_HIDE 0x40000000
|
||||||
@ -92,7 +108,7 @@ inline fastcall dword WaitEventTimeout(EBX)
|
|||||||
$int 0x40
|
$int 0x40
|
||||||
}
|
}
|
||||||
|
|
||||||
inline fastcall SetEventMask(EBX)
|
inline fastcall dword SetEventMask(EBX)
|
||||||
{
|
{
|
||||||
$mov eax,40
|
$mov eax,40
|
||||||
$int 0x40
|
$int 0x40
|
||||||
|
@ -121,4 +121,16 @@ dword load_image(dword filename)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DrawLibImage(dword image_pointer,x,y,w,h,offx,offy) {
|
||||||
|
img_draw stdcall (
|
||||||
|
image_pointer,
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
w,
|
||||||
|
h,
|
||||||
|
offx,
|
||||||
|
offy
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
Loading…
Reference in New Issue
Block a user