mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-20 22:26:54 +01:00
Work (but don't call into java world) when initialized with a null
instance of UtilCtxt; implement util_makeEmptyDict().
This commit is contained in:
parent
69ce7e2ef0
commit
67cb2421c9
2 changed files with 66 additions and 34 deletions
|
@ -22,6 +22,7 @@
|
||||||
#include <jni.h>
|
#include <jni.h>
|
||||||
|
|
||||||
#include "utilwrapper.h"
|
#include "utilwrapper.h"
|
||||||
|
#include "anddict.h"
|
||||||
#include "andutils.h"
|
#include "andutils.h"
|
||||||
#include "LocalizedStrIncludes.h"
|
#include "LocalizedStrIncludes.h"
|
||||||
|
|
||||||
|
@ -33,7 +34,7 @@ typedef struct _TimerStorage {
|
||||||
typedef struct _AndUtil {
|
typedef struct _AndUtil {
|
||||||
XW_UtilCtxt util;
|
XW_UtilCtxt util;
|
||||||
JNIEnv** env;
|
JNIEnv** env;
|
||||||
jobject j_util; /* global ref to object implementing XW_UtilCtxt */
|
jobject jutil; /* global ref to object implementing XW_UtilCtxt */
|
||||||
TimerStorage timerStorage[NUM_TIMERS_PLUS_ONE];
|
TimerStorage timerStorage[NUM_TIMERS_PLUS_ONE];
|
||||||
XP_UCHAR* userStrings[N_AND_USER_STRINGS];
|
XP_UCHAR* userStrings[N_AND_USER_STRINGS];
|
||||||
} AndUtil;
|
} AndUtil;
|
||||||
|
@ -63,33 +64,44 @@ and_util_makeStreamFromAddr( XW_UtilCtxt* uc, XP_PlayerAddr channelNo )
|
||||||
#define UTIL_CBK_HEADER(nam,sig) \
|
#define UTIL_CBK_HEADER(nam,sig) \
|
||||||
AndUtil* util = (AndUtil*)uc; \
|
AndUtil* util = (AndUtil*)uc; \
|
||||||
JNIEnv* env = *util->env; \
|
JNIEnv* env = *util->env; \
|
||||||
jmethodID mid = getMethodID( env, util->j_util, nam, sig )
|
if ( NULL != util->jutil ) { \
|
||||||
|
jmethodID mid = getMethodID( env, util->jutil, nam, sig )
|
||||||
|
|
||||||
|
#define UTIL_CBK_TAIL() \
|
||||||
|
} else { \
|
||||||
|
XP_LOGF( "%s: skipping call into java because jutil==NULL", \
|
||||||
|
__func__ ); \
|
||||||
|
}
|
||||||
|
|
||||||
static XWBonusType and_util_getSquareBonus( XW_UtilCtxt* uc,
|
static XWBonusType and_util_getSquareBonus( XW_UtilCtxt* uc,
|
||||||
const ModelCtxt* XP_UNUSED(model),
|
const ModelCtxt* XP_UNUSED(model),
|
||||||
XP_U16 col, XP_U16 row )
|
XP_U16 col, XP_U16 row )
|
||||||
{
|
{
|
||||||
|
XWBonusType result = BONUS_NONE;
|
||||||
UTIL_CBK_HEADER("getSquareBonus","(II)I" );
|
UTIL_CBK_HEADER("getSquareBonus","(II)I" );
|
||||||
return (*env)->CallIntMethod( env, util->j_util, mid,
|
result = (*env)->CallIntMethod( env, util->jutil, mid,
|
||||||
col, row );
|
col, row );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
and_util_userError( XW_UtilCtxt* uc, UtilErrID id )
|
and_util_userError( XW_UtilCtxt* uc, UtilErrID id )
|
||||||
{
|
{
|
||||||
UTIL_CBK_HEADER( "userError", "(I)V" );
|
UTIL_CBK_HEADER( "userError", "(I)V" );
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid, id );
|
(*env)->CallVoidMethod( env, util->jutil, mid, id );
|
||||||
if ((*env)->ExceptionOccurred(env)) {
|
if ((*env)->ExceptionOccurred(env)) {
|
||||||
(*env)->ExceptionDescribe(env);
|
(*env)->ExceptionDescribe(env);
|
||||||
(*env)->ExceptionClear(env);
|
(*env)->ExceptionClear(env);
|
||||||
XP_LOGF( "exception found" );
|
XP_LOGF( "exception found" );
|
||||||
}
|
}
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
}
|
}
|
||||||
|
|
||||||
static XP_Bool
|
static XP_Bool
|
||||||
and_util_userQuery( XW_UtilCtxt* uc, UtilQueryID id, XWStreamCtxt* stream )
|
and_util_userQuery( XW_UtilCtxt* uc, UtilQueryID id, XWStreamCtxt* stream )
|
||||||
{
|
{
|
||||||
|
jboolean result = XP_FALSE;
|
||||||
XP_ASSERT( id < QUERY_LAST_COMMON );
|
XP_ASSERT( id < QUERY_LAST_COMMON );
|
||||||
UTIL_CBK_HEADER("userQuery", "(ILjava/lang/String;)Z" );
|
UTIL_CBK_HEADER("userQuery", "(ILjava/lang/String;)Z" );
|
||||||
|
|
||||||
|
@ -97,11 +109,11 @@ and_util_userQuery( XW_UtilCtxt* uc, UtilQueryID id, XWStreamCtxt* stream )
|
||||||
if ( NULL != stream ) {
|
if ( NULL != stream ) {
|
||||||
jstr = streamToJString( MPPARM(util->util.mpool) env, stream );
|
jstr = streamToJString( MPPARM(util->util.mpool) env, stream );
|
||||||
}
|
}
|
||||||
jboolean result = (*env)->CallBooleanMethod( env, util->j_util, mid,
|
result = (*env)->CallBooleanMethod( env, util->jutil, mid, id, jstr );
|
||||||
id, jstr );
|
|
||||||
if ( NULL != jstr ) {
|
if ( NULL != jstr ) {
|
||||||
(*env)->DeleteLocalRef( env, jstr );
|
(*env)->DeleteLocalRef( env, jstr );
|
||||||
}
|
}
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,8 +121,8 @@ static XP_S16
|
||||||
and_util_userPickTile( XW_UtilCtxt* uc, const PickInfo* pi,
|
and_util_userPickTile( XW_UtilCtxt* uc, const PickInfo* pi,
|
||||||
XP_U16 playerNum, const XP_UCHAR** texts, XP_U16 nTiles )
|
XP_U16 playerNum, const XP_UCHAR** texts, XP_U16 nTiles )
|
||||||
{
|
{
|
||||||
UTIL_CBK_HEADER("userPickTile", "(I[Ljava/lang/String;)I" );
|
|
||||||
XP_S16 result = -1;
|
XP_S16 result = -1;
|
||||||
|
UTIL_CBK_HEADER("userPickTile", "(I[Ljava/lang/String;)I" );
|
||||||
|
|
||||||
#ifdef FEATURE_TRAY_EDIT
|
#ifdef FEATURE_TRAY_EDIT
|
||||||
++error; /* need to pass pi if this is on */
|
++error; /* need to pass pi if this is on */
|
||||||
|
@ -118,11 +130,11 @@ and_util_userPickTile( XW_UtilCtxt* uc, const PickInfo* pi,
|
||||||
|
|
||||||
jobject jtexts = makeStringArray( env, nTiles, texts );
|
jobject jtexts = makeStringArray( env, nTiles, texts );
|
||||||
|
|
||||||
result = (*env)->CallIntMethod( env, util->j_util, mid,
|
result = (*env)->CallIntMethod( env, util->jutil, mid,
|
||||||
playerNum, jtexts );
|
playerNum, jtexts );
|
||||||
|
|
||||||
(*env)->DeleteLocalRef( env, jtexts );
|
(*env)->DeleteLocalRef( env, jtexts );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
return result;
|
return result;
|
||||||
} /* and_util_userPickTile */
|
} /* and_util_userPickTile */
|
||||||
|
|
||||||
|
@ -131,7 +143,6 @@ static XP_Bool
|
||||||
and_util_askPassword( XW_UtilCtxt* uc, const XP_UCHAR* name,
|
and_util_askPassword( XW_UtilCtxt* uc, const XP_UCHAR* name,
|
||||||
XP_UCHAR* buf, XP_U16* len )
|
XP_UCHAR* buf, XP_U16* len )
|
||||||
{
|
{
|
||||||
LOG_FUNC();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -139,7 +150,6 @@ static void
|
||||||
and_util_trayHiddenChange(XW_UtilCtxt* uc, XW_TrayVisState newState,
|
and_util_trayHiddenChange(XW_UtilCtxt* uc, XW_TrayVisState newState,
|
||||||
XP_U16 nVisibleRows )
|
XP_U16 nVisibleRows )
|
||||||
{
|
{
|
||||||
LOG_FUNC();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -149,9 +159,9 @@ and_util_yOffsetChange(XW_UtilCtxt* uc, XP_U16 oldOffset, XP_U16 newOffset )
|
||||||
AndUtil* util = (AndUtil*)uc;
|
AndUtil* util = (AndUtil*)uc;
|
||||||
JNIEnv* env = *util->env;
|
JNIEnv* env = *util->env;
|
||||||
const char* sig = "(II)V";
|
const char* sig = "(II)V";
|
||||||
jmethodID mid = getMethodID( env, util->j_util, "yOffsetChange", sig );
|
jmethodID mid = getMethodID( env, util->jutil, "yOffsetChange", sig );
|
||||||
|
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid,
|
(*env)->CallVoidMethod( env, util->jutil, mid,
|
||||||
oldOffset, newOffset );
|
oldOffset, newOffset );
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -168,7 +178,8 @@ static void
|
||||||
and_util_notifyGameOver( XW_UtilCtxt* uc )
|
and_util_notifyGameOver( XW_UtilCtxt* uc )
|
||||||
{
|
{
|
||||||
UTIL_CBK_HEADER( "notifyGameOver", "()V" );
|
UTIL_CBK_HEADER( "notifyGameOver", "()V" );
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid );
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -182,8 +193,11 @@ and_util_hiliteCell( XW_UtilCtxt* uc, XP_U16 col, XP_U16 row )
|
||||||
static XP_Bool
|
static XP_Bool
|
||||||
and_util_engineProgressCallback( XW_UtilCtxt* uc )
|
and_util_engineProgressCallback( XW_UtilCtxt* uc )
|
||||||
{
|
{
|
||||||
|
XP_Bool result = XP_FALSE;
|
||||||
UTIL_CBK_HEADER("engineProgressCallback","()Z" );
|
UTIL_CBK_HEADER("engineProgressCallback","()Z" );
|
||||||
return (*env)->CallBooleanMethod( env, util->j_util, mid );
|
result = (*env)->CallBooleanMethod( env, util->jutil, mid );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is added for java, not part of the util api */
|
/* This is added for java, not part of the util api */
|
||||||
|
@ -206,15 +220,17 @@ and_util_setTimer( XW_UtilCtxt* uc, XWTimerReason why, XP_U16 when,
|
||||||
TimerStorage* storage = &util->timerStorage[why];
|
TimerStorage* storage = &util->timerStorage[why];
|
||||||
storage->proc = proc;
|
storage->proc = proc;
|
||||||
storage->closure = closure;
|
storage->closure = closure;
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid,
|
(*env)->CallVoidMethod( env, util->jutil, mid,
|
||||||
why, when, (int)storage );
|
why, when, (int)storage );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
and_util_clearTimer( XW_UtilCtxt* uc, XWTimerReason why )
|
and_util_clearTimer( XW_UtilCtxt* uc, XWTimerReason why )
|
||||||
{
|
{
|
||||||
UTIL_CBK_HEADER("clearTimer", "(I)V" );
|
UTIL_CBK_HEADER("clearTimer", "(I)V" );
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid, why );
|
(*env)->CallVoidMethod( env, util->jutil, mid, why );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -222,7 +238,8 @@ static void
|
||||||
and_util_requestTime( XW_UtilCtxt* uc )
|
and_util_requestTime( XW_UtilCtxt* uc )
|
||||||
{
|
{
|
||||||
UTIL_CBK_HEADER("requestTime", "()V" );
|
UTIL_CBK_HEADER("requestTime", "()V" );
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid );
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
}
|
}
|
||||||
|
|
||||||
static XP_Bool
|
static XP_Bool
|
||||||
|
@ -245,19 +262,24 @@ and_util_getCurSeconds( XW_UtilCtxt* uc )
|
||||||
static DictionaryCtxt*
|
static DictionaryCtxt*
|
||||||
and_util_makeEmptyDict( XW_UtilCtxt* uc )
|
and_util_makeEmptyDict( XW_UtilCtxt* uc )
|
||||||
{
|
{
|
||||||
LOG_FUNC();
|
AndUtil* util = (AndUtil*)uc;
|
||||||
|
#ifdef STUBBED_DICT
|
||||||
|
XP_ASSERT(0);
|
||||||
|
#else
|
||||||
|
return and_dictionary_make_empty( MPPARM_NOCOMMA( util->util.mpool ) );
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static const XP_UCHAR*
|
static const XP_UCHAR*
|
||||||
and_util_getUserString( XW_UtilCtxt* uc, XP_U16 stringCode )
|
and_util_getUserString( XW_UtilCtxt* uc, XP_U16 stringCode )
|
||||||
{
|
{
|
||||||
|
XP_UCHAR* result = NULL;
|
||||||
UTIL_CBK_HEADER("getUserString", "(I)Ljava/lang/String;" );
|
UTIL_CBK_HEADER("getUserString", "(I)Ljava/lang/String;" );
|
||||||
int index = stringCode - 1; /* see LocalizedStrIncludes.h */
|
int index = stringCode - 1; /* see LocalizedStrIncludes.h */
|
||||||
XP_ASSERT( index < VSIZE( util->userStrings ) );
|
XP_ASSERT( index < VSIZE( util->userStrings ) );
|
||||||
|
|
||||||
if ( ! util->userStrings[index] ) {
|
if ( ! util->userStrings[index] ) {
|
||||||
jstring jresult = (*env)->CallObjectMethod( env, util->j_util, mid,
|
jstring jresult = (*env)->CallObjectMethod( env, util->jutil, mid,
|
||||||
stringCode );
|
stringCode );
|
||||||
jsize len = (*env)->GetStringUTFLength( env, jresult );
|
jsize len = (*env)->GetStringUTFLength( env, jresult );
|
||||||
XP_UCHAR* buf = XP_MALLOC( util->util.mpool, len + 1 );
|
XP_UCHAR* buf = XP_MALLOC( util->util.mpool, len + 1 );
|
||||||
|
@ -272,7 +294,9 @@ and_util_getUserString( XW_UtilCtxt* uc, XP_U16 stringCode )
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG_RETURNF( "%s", util->userStrings[index] );
|
LOG_RETURNF( "%s", util->userStrings[index] );
|
||||||
return util->userStrings[index];
|
result = util->userStrings[index];
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -288,7 +312,8 @@ static void
|
||||||
and_util_remSelected(XW_UtilCtxt* uc)
|
and_util_remSelected(XW_UtilCtxt* uc)
|
||||||
{
|
{
|
||||||
UTIL_CBK_HEADER("remSelected", "()V" );
|
UTIL_CBK_HEADER("remSelected", "()V" );
|
||||||
(*env)->CallVoidMethod( env, util->j_util, mid );
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -335,9 +360,10 @@ and_util_makeJBitmap( XW_UtilCtxt* uc, int nCols, int nRows,
|
||||||
"(II[Z)Landroid/graphics/drawable/BitmapDrawable;" );
|
"(II[Z)Landroid/graphics/drawable/BitmapDrawable;" );
|
||||||
jbooleanArray jcolors = makeBooleanArray( env, nCols*nRows, colors );
|
jbooleanArray jcolors = makeBooleanArray( env, nCols*nRows, colors );
|
||||||
|
|
||||||
bitmap = (*env)->CallObjectMethod( env, util->j_util, mid,
|
bitmap = (*env)->CallObjectMethod( env, util->jutil, mid,
|
||||||
nCols, nRows, jcolors );
|
nCols, nRows, jcolors );
|
||||||
(*env)->DeleteLocalRef( env, jcolors );
|
(*env)->DeleteLocalRef( env, jcolors );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
return bitmap;
|
return bitmap;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -353,22 +379,24 @@ and_util_splitFaces( XW_UtilCtxt* uc, const XP_U8* bytes, jsize len )
|
||||||
XP_MEMCPY( jp, bytes, len );
|
XP_MEMCPY( jp, bytes, len );
|
||||||
(*env)->ReleaseByteArrayElements( env, jbytes, jp, 0 );
|
(*env)->ReleaseByteArrayElements( env, jbytes, jp, 0 );
|
||||||
|
|
||||||
strarray = (*env)->CallObjectMethod( env, util->j_util, mid, jbytes );
|
strarray = (*env)->CallObjectMethod( env, util->jutil, mid, jbytes );
|
||||||
(*env)->DeleteLocalRef( env, jbytes );
|
(*env)->DeleteLocalRef( env, jbytes );
|
||||||
|
UTIL_CBK_TAIL();
|
||||||
return strarray;
|
return strarray;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
XW_UtilCtxt*
|
XW_UtilCtxt*
|
||||||
makeUtil( MPFORMAL JNIEnv** envp, jobject j_util, CurGameInfo* gi,
|
makeUtil( MPFORMAL JNIEnv** envp, jobject jutil, CurGameInfo* gi,
|
||||||
AndGlobals* closure )
|
AndGlobals* closure )
|
||||||
{
|
{
|
||||||
AndUtil* util = (AndUtil*)XP_CALLOC( mpool, sizeof(*util) );
|
AndUtil* util = (AndUtil*)XP_CALLOC( mpool, sizeof(*util) );
|
||||||
UtilVtable* vtable = (UtilVtable*)XP_CALLOC( mpool, sizeof(*vtable) );
|
UtilVtable* vtable = (UtilVtable*)XP_CALLOC( mpool, sizeof(*vtable) );
|
||||||
util->env = envp;
|
util->env = envp;
|
||||||
JNIEnv* env = *envp;
|
JNIEnv* env = *envp;
|
||||||
util->j_util = (*env)->NewGlobalRef( env, j_util );
|
if ( NULL != jutil ) {
|
||||||
|
util->jutil = (*env)->NewGlobalRef( env, jutil );
|
||||||
|
}
|
||||||
util->util.vtable = vtable;
|
util->util.vtable = vtable;
|
||||||
MPASSIGN( util->util.mpool, mpool );
|
MPASSIGN( util->util.mpool, mpool );
|
||||||
util->util.closure = closure;
|
util->util.closure = closure;
|
||||||
|
@ -387,7 +415,7 @@ makeUtil( MPFORMAL JNIEnv** envp, jobject j_util, CurGameInfo* gi,
|
||||||
SET_PROC(trayHiddenChange);
|
SET_PROC(trayHiddenChange);
|
||||||
SET_PROC(yOffsetChange);
|
SET_PROC(yOffsetChange);
|
||||||
#ifdef XWFEATURE_TURNCHANGENOTIFY
|
#ifdef XWFEATURE_TURNCHANGENOTIFY
|
||||||
SET_PROC( turnChanged);
|
SET_PROC( turnChanged);
|
||||||
#endif
|
#endif
|
||||||
SET_PROC(notifyGameOver);
|
SET_PROC(notifyGameOver);
|
||||||
SET_PROC(hiliteCell);
|
SET_PROC(hiliteCell);
|
||||||
|
@ -413,14 +441,14 @@ SET_PROC( turnChanged);
|
||||||
SET_PROC(engineStopping);
|
SET_PROC(engineStopping);
|
||||||
#endif
|
#endif
|
||||||
#undef SET_PROC
|
#undef SET_PROC
|
||||||
|
|
||||||
return (XW_UtilCtxt*)util;
|
return (XW_UtilCtxt*)util;
|
||||||
} /* makeUtil */
|
} /* makeUtil */
|
||||||
|
|
||||||
void
|
void
|
||||||
destroyUtil( XW_UtilCtxt* utilc )
|
destroyUtil( XW_UtilCtxt** utilc )
|
||||||
{
|
{
|
||||||
AndUtil* util = (AndUtil*)utilc;
|
LOG_FUNC();
|
||||||
|
AndUtil* util = (AndUtil*)*utilc;
|
||||||
JNIEnv *env = *util->env;
|
JNIEnv *env = *util->env;
|
||||||
|
|
||||||
int ii;
|
int ii;
|
||||||
|
@ -431,7 +459,11 @@ destroyUtil( XW_UtilCtxt* utilc )
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
(*env)->DeleteGlobalRef( env, util->j_util );
|
if ( NULL != util->jutil ) {
|
||||||
|
(*env)->DeleteGlobalRef( env, util->jutil );
|
||||||
|
}
|
||||||
XP_FREE( util->util.mpool, util->util.vtable );
|
XP_FREE( util->util.mpool, util->util.vtable );
|
||||||
XP_FREE( util->util.mpool, util );
|
XP_FREE( util->util.mpool, util );
|
||||||
|
*utilc = NULL;
|
||||||
|
LOG_RETURN_VOID();
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
|
|
||||||
XW_UtilCtxt* makeUtil( MPFORMAL JNIEnv** env, jobject j_util,
|
XW_UtilCtxt* makeUtil( MPFORMAL JNIEnv** env, jobject j_util,
|
||||||
CurGameInfo* gi, AndGlobals* globals );
|
CurGameInfo* gi, AndGlobals* globals );
|
||||||
void destroyUtil( XW_UtilCtxt* util );
|
void destroyUtil( XW_UtilCtxt** util );
|
||||||
|
|
||||||
bool utilTimerFired( XW_UtilCtxt* util, XWTimerReason why, int handle );
|
bool utilTimerFired( XW_UtilCtxt* util, XWTimerReason why, int handle );
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue