/*<Header>==============================================================
*
*	アイコン時計 / "MAIN.C"
*
*		[ EIN(tm) project : サンプルプログラム ]
*
*	COPYRIGHT  Nam  1994, All rights reserved.
*
*	配付・組込・改変・商利用すべて自由。ただし無保証っす
*
*-----------------------------------------------------------------------
*	V1.0L01α	94.07.07/Nam	プロトタイプ
*	V1.0L30		94.07.19/Nam	EIN(TM)公開版
*	V1.0L31		94.09.05/Nam	GM_QUIT時に自分でWINDOWを消去
*</Header>==============================================================*/
#include	<stdio.h>
#include	<stdlib.h>
#include	<string.h>
#include	<snd.h>
#include	<winb.h>
#include	<te.h>
#include	<fntb.h>
#include	<gui.h>
#include	<egb.h>
//#include	<file_dlg.h>
//#include	<tifflib.h>
#include	<msdos.cf>
#include	<loader.h>
#include	<math.h>
#include	<io.h>
#include	<guidbg.h>
#include	<wgb.h>
// ※ ここまでが平均的に使用されるヘッダ

#include	<time.h>		// このサンプルでのみ必要
#include	<mos.h>			// このサンプルでのみ必要

#include	"..\ein\eintm.h"	// EIN(TM)関連ライブラリのヘッダ


#define	ERROR	(-1)
#define	PSET	(0)


int		ID_ICON[1];					// ウィンドゥ部品のID

/*= リジューム用設定=================================================*/
FRAME	RSM_ICONFR;
char	RSMID[]="%%% CLocK icon %%%";
char	RSMTITLE[]="アイコン時計  V0.1 L31";
/*===================================================================*/

char	GV_workBuf[32*32*4] ;		// パターン書換え用ワークエリア
int		ndlCol[3];					// 時計の針の色
int		updateFlag = ERROR;			// 画面更新中か否かを示すフラグ

char	*GV_egbWork ;
int		GV_apliLot ;
void	( *GV_defIdleTask )() ;


char	*guiEgbPtr ;				//	EGB のワークアドレス
MMICTRL	mmic = {
				 SCREEN16 | SCREENIGNORE,		//	resolution
				 SCREENUNUSED,
				 0, SCREENAVAILABLE, 0,	//	write_page,display_page,priority
				 SCREENAVAILABLE,	//	mode
				 SCREENEXPAND,		//	vram_x
				 0,0,0,0,			//	size,*ptr,asize,*atpr
				 0,0,0,0,			//	frame
				 -32767,			//	move
				 -32767,
				  32767,
				  32767,
				 15,8,7,15			//	color
				} ;

/*===================================================================*/
/*  メインループ                                                     */
/*===================================================================*/
void main()
{
	extern int APL_init() ;
	extern int eraseWin();

	int		kobj;

	// 初期化処理
	if (MMI_Open( &mmic ) == NOERR){
		// 二重起動のcheck
		if ( (kobj = MMI_CallMessage( MMI_GetApliId(), GM_QUERYID, QM_KIND, 1)) > NOERR ){
			MMI_CallMessage( MMI_GetApliId(), GM_SWITCH, FALSE, kobj );
		} else {
			// アイドルタスクに登録
			setIdleTask();
			// 初期化に成功すればメインループに
			if ( APL_init()==NOERR ){
				// メインループ
				MMI_ExecSystem();
			}
			// アイドルの解除
			resetIdleTask();
			// ウィンドゥ消去(※SHELLに消去させると遅いので)
			eraseWin();
		}
	}
	// 終了処理
	MMI_Close() ;
}

/*===================================================================*/
/*  初期化処理                                                       */
/*===================================================================*/
int APL_init()
{
	extern	MMIINIT	initDataRES_ICON ;

//	extern	int ICONmaxId;
//	extern	unsigned char *ICONTbl[];

	extern int	userFunc();
	extern int updateResolution();

	HYPER	hyp;
	register int	ret ;


	// EGB ワークアドレスを取得し、ローカルワークにコピー
	guiEgbPtr = MMI_GetEgbPtr() ;
	if ( (GV_apliLot = TL_getLot()) < 1 ){
		return ERROR ;
	}
	if( (GV_egbWork = TL_mallocMemory( GV_apliLot, EgbWorkSize )) == NULL ){
		return ERROR ;
	}
	FM_memcpy( GV_egbWork, guiEgbPtr, EgbWorkSize ) ;

	// GUI部品の初期化
	if ((ret = MMI_initHyper()) < 0)			//	ハイパー型
		return ret;
	if ((ret = MMI_initDialogL40()) < 0)		//	ダイアログ型
		return ret;
	if ((ret = MMI_initWindowL40()) < 0)		//	ウィンドウ型
		return ret;

	// リソースを登録
	if ((ret = MMI_Init(&initDataRES_ICON)) < 0)
		return ret ;

//	// ユーザアイコンを登録
//	MMI_SetIconTable( ICONTbl, ICONmaxId ) ;

	// GUI色をメニュー色に設定
	EIN_initGuiColor();
	// 仮想画面情報を更新
	updateResolution();

	// ウィンドウ類をベース部品にATTACH
	MMI_SendMessage( ID_ICON[0], MM_ATTACH, 1, MMI_GetBaseObj() ) ;

	// スイッチャー対応関数の登録
	MMI_SendMessage( MMI_GetBaseObj(), MM_SETEXEC, 1, userFunc ) ;

	// タスクリストにタイトルを登録
	MMI_CallMessage( MMI_GetApliId(), GM_TITLE, (int)RSMTITLE, 0 ) ;

	/*----------------------------------------------------------------*/
	/*  リジューム読み込み (※のついた処理は必ず行ってください)       */
	/*----------------------------------------------------------------*/
	char	work[RSMWORKSIZE];	// リジュームマネージャのワークエリア
	char	buf[1024];
	char	*ptr;
	int		size, x, y;

	// ※リジュームマネージャの初期化(アプリのIDタグ登録)
	EIN_rsmInit( work, RSMID );
	// ※読み込みバッファ設定
	EIN_rsmBufSet( work, buf, 1023 );
	// ※リジューム情報読み込み
	size = EIN_rsmLoad( work );
	#ifdef DEBUG
	printf(" resume size(%d)\n",size);
	#endif
	if ( size > 0 ){
		// ICON表示位置
		if ( ((ptr = strstr(buf,"\nICON: ")) != NULL ) &&
			 (sscanf(ptr,"\nICON: %d %d\n", &x, &y) > 1 ) ){
			// 枠座標を得る
			MMI_SendMessage( ID_ICON[0], MM_GETHYPER, 1, &hyp );
			RSM_ICONFR.lupx = x;
			RSM_ICONFR.lupy = y;
			RSM_ICONFR.rdwx = x+(hyp.fr.rdwx-hyp.fr.lupx);
			RSM_ICONFR.rdwy = y+(hyp.fr.rdwy-hyp.fr.lupy);
			#ifdef DEBUG
			printf("ICON: move to (%d,%d)\n", x, y);
			#endif
			if ( x<-31 ){
				x=0;
			}
			if ( y<-31 ){
				y=0;
			}
			MMI_SendMessage( ID_ICON[0], MM_MOVE, 1, &RSM_ICONFR ) ;
		}
	} else {
		#ifdef DEBUG
		printf("Can't find resume file.\n");
		#endif
	}

	//	背景を表示
	MMI_SendMessage( MMI_GetBaseObj(), MM_SHOW, 0 ) ;

	return NOERR ;
}

/*===================================================================*/
/* リジューム情報更新 (※がついた処理は必ず行って下さい)             */
/*===================================================================*/
void	SaveResumeFile()
{
	char	work[RSMWORKSIZE];
	char	buf[1024];
	char	aplpath[128];
	HYPER	hyp1;
	register int ret;
	
	// 枠座標を得る
	MMI_SendMessage( ID_ICON[0], MM_GETHYPER, 1, &hyp1 );
	
	// 移動していたらセーブ
	if ( (RSM_ICONFR.lupx  != hyp1.fr.lupx) ||
	     (RSM_ICONFR.lupy  != hyp1.fr.lupy) ){
		// ※マネージャ初期化
		EIN_rsmInit( work, RSMID );
		// ※バッファ設定(アプリ識別ID,更新時刻は自動設定されます)
		EIN_rsmBufSet( work, buf, 1023 );
		// ※アプリディレクトリ保存
		MMI_CallMessage( MMI_GetApliId(), GM_QUERYID, QM_PATH, (int)aplpath );
		EIN_rsmBufPrintf( work, "APLPATH: %s", aplpath );
		// ※アプリ名保存
		EIN_rsmBufPrintf( work, "TITLE:   %s", RSMTITLE );
		// ※コメント保存
		EIN_rsmBufCat( work, "COMMENT: EIN(tm)のサンプルです。");
		// アイコン位置保存
		EIN_rsmBufPrintf( work, "ICON: %d %d", hyp1.fr.lupx, hyp1.fr.lupy );
		// ※バッファ終端(^A)設定
		EIN_rsmBufTail( work );
		// ※バッファをセーブ
		ret = EIN_rsmSave( work );
		#ifdef DEBUG
		printf(" resume save size(%d)   ptr:(%d,%d)\n",ret, hyp.fr.lupx, hyp.fr.lupy);
		#endif
		RSM_ICONFR.lupx = hyp1.fr.lupx;
		RSM_ICONFR.lupy = hyp1.fr.lupy;
		RSM_ICONFR.rdwx = hyp1.fr.rdwx;
		RSM_ICONFR.rdwy = hyp1.fr.rdwy;
	}
}

/*===================================================================*/
/*  イベントループ関数                                               */
/*===================================================================*/
int		userFunc( int apliId, int messId, int info, int data )
{
//	extern int updateResolution();
	extern int startUpdate();

	register	int		ret ;

	ret = ILLEGAL_FUNCTION ;
	switch( messId ){
		//	他タスクからスイッチしてきた
		case GM_WAKE :
			break ;

		//	他タスクにスイッチする
		case GM_SLEEP :
			break ;

		//	画面変更前処理
		case GM_PRESCRCHG :
			updateFlag = ERROR;		// 画面更新禁止
			break ;

		//	画面変更後処理
		case GM_POSTSCRCHG :
//			updateResolution();
//			startUpdate();
			/*=======================================================*/
			/*  本来はこの処理中で画面解像度を得られるはずなのですが */
			/*  どうもうまく処理されません(T-T)                      */
			/*  仕方無いので、解像度取得・変更関数をイベントに積んで */
			/*  関数終了後に呼ぶようにしました。                     */
			/*  （この方法を教えてくれた、ま〜おじさんに感謝!）      */
			/*=======================================================*/
			static int	argv[2] = { 1, 1 } ;
			EVENT	ev ;
			ev.what = EVEXEC ;
			ev.shift = 0 ;
			ev.info = (int)startUpdate;
			ev.data = (int)argv ;
			MMI_SetEvnt(&ev) ;
			break ;

		//	一時停止要求が出た
		case GM_PAUSE :
			updateFlag = ERROR;
			break ;

		//	一時停止解除要求が出た
		case GM_CONTINUE :
			updateFlag = NOERR;
			break ;

		//	終了要求が出た
		case GM_QUIT :
			// リジューム
			SaveResumeFile();
			//	正常終了
			MMI_SetHaltFlag( TRUE ) ;	//	イベントループを終了させるフラグ
			ret = NOERR ;
			break ;

	}

	return ret ;
}

/*===================================================================*/
/*  アイドルタスク関数(ウィンドゥ上に時計表示)                       */
/*===================================================================*/
void	userIdleTask()
{
	#define	PI_B_2	(3.14159265358979323846*2.0)	/* PI×2 */
	#define	PI_D_2	(3.14159265358979323846/2.0)	/* PI÷2 */

	typedef	struct {
		_Far char *pixel;
		short	x1,y1;
		short	x2,y2;
		} EGB_BLOCK;

	typedef	struct {
		short	sum;
		short	x1,y1;
		short	x2,y2;
		} EGB_LINE;

	static int	oldsec = -1;
	
	EGB_BLOCK	put;
	EGB_LINE	line;
	HYPER	hyp ;
	int		hour, min, sec, ox, oy;
	double	rm, r, th;
	int		btn, bx, by;
	Rect	rmos;
	time_t		ct = time(NULL);
	struct tm  *lt = localtime(&ct);
	char		cl[EINCLIPSIZE];

	// SCREENCHANGE中なので更新しない
	if ( updateFlag != NOERR ){
		return;
	}
	hour = lt->tm_hour;
	min  = lt->tm_min;
	sec  = lt->tm_sec;
	
	// まだ一秒経っていないなら終了
	if ( sec == oldsec ){
		return;
	}
	oldsec = sec;
	
	// 書込先の座標を得る
	MMI_SendMessage( ID_ICON[0], MM_GETHYPER, 1, &hyp );
	ox = (hyp.fr.rdwx - hyp.fr.lupx -2) / 2;
	oy = (hyp.fr.rdwy - hyp.fr.lupy -2) / 2;
	rm = (ox + oy) / 2;
	
	// 仮想画面消去
	EGB_writePage( GV_egbWork, 0x80 ) ;
	EGB_clearScreen( GV_egbWork ) ;

	/* 短針 */
	EGB_color( GV_egbWork, 0, ndlCol[2] );
	EGB_penSize( GV_egbWork, 3 );
	r = (double)rm * 0.8;
	th = -(PI_B_2 * (((((double)(hour % 12)) * 3600.0 + ((double)min) * 60.0)) / (12.0 * 3600.0)) - PI_D_2);
	line.sum= 2;
	line.x1 = ox;
	line.y1 = oy;
	line.x2 = ox + (int)(r * cos(th));
	line.y2 = oy - (int)(r * sin(th));
	EGB_connect( GV_egbWork, (char *)&line );
	/* 長針 */
	EGB_color( GV_egbWork, 0, ndlCol[1] );
	EGB_penSize( GV_egbWork, 2 );
	r = (double)rm;
	th = -(PI_B_2 * ((((double)min) * 60.0 + (double)sec) / (60.0 * 60.0)) - PI_D_2);
//	line.sum= 2;
//	line.x1 = ox;
//	line.y1 = oy;
	line.x2 = ox + (int)(r * cos(th));
	line.y2 = oy - (int)(r * sin(th));
	EGB_connect( GV_egbWork, (char *)&line );
	/* 秒針 */
	EGB_color( GV_egbWork, 0, ndlCol[0] );
	EGB_penSize( GV_egbWork, 1 );
	r = (double)rm * 0.9;
	th = -(PI_B_2 * (((double)sec) / 60.0) - PI_D_2);
//	line.sum= 2;
//	line.x1 = ox;
//	line.y1 = oy;
	line.x2 = ox + (int)(r * cos(th));
	line.y2 = oy - (int)(r * sin(th));
	EGB_connect( GV_egbWork, (char *)&line );

	// クリップ
	//	原点移動を戻し、描画エリアにクリップをかける
	EIN_clipOpen( cl );
	put.pixel = GV_workBuf ;
	put.x1    = hyp.fr.lupx+2;
	put.y1    = hyp.fr.lupy+2;
	put.x2    = hyp.fr.rdwx-2;
	put.y2    = hyp.fr.rdwy-2;
	EGB_writeMode( guiEgbPtr, PSET ) ;
	// 現在のマウスカーソルの位置を読む
	MOS_rdpos( &btn, &bx, &by );
	rmos.left = bx;
	rmos.up   = by;
	rmos.right= rmos.left + 32;
	rmos.down = rmos.up   + 32;
	
	if ( SectRect( &rmos, (Rect *)&hyp.fr ) ){
		// 内側なのでマウスを消そうね
		MG_mosDisp(2);
		WGB_putBlock( guiEgbPtr, 3, (char *)&put ) ;
		MG_mosDisp(3);
	} else {
		// 外側
		WGB_putBlock( guiEgbPtr, 3, (char *)&put ) ;
	}
	// クリップ解除
	EIN_clipClose( cl );
}

/*===================================================================*/
/*  アイドルタスクに登録                                             */
/*===================================================================*/
int		setIdleTask()
{
	void	( *idleTask )() ;

	/*	アイドル処理関数の取得	*/
	idleTask = MMI_GetIdleTaskFunc() ;

	/*	二重登録禁止チェック	*/
	if( idleTask != userIdleTask ){
		GV_defIdleTask = idleTask ;
		MMI_SetIdleTaskFunc( (void ((*)(void)))userIdleTask );
	}

	return ERROR ;
}

/*===================================================================*/
/*  アイドルタスクから削除                                           */
/*===================================================================*/
int		resetIdleTask()
{
	void	( *IdleTask )() ;

	/*	アイドル処理関数の取得	*/
	IdleTask = MMI_GetIdleTaskFunc() ;

	/*	USER関数登録のチェック	*/
	if( IdleTask == userIdleTask ){
		MMI_SetIdleTaskFunc( GV_defIdleTask ) ;
		GV_defIdleTask = 0 ;
	}

	return ERROR ;
}

/*===================================================================*/
/*  現在の画面モードに応じて仮想画面再設定                           */
/*===================================================================*/
int		updateResolution()
{
	int		modebuf[8];
	int		vbits;
	int		clsColor;
	
	// EGB 画面情報取得
	EGB_getResolutionPage( 0, modebuf );
	vbits = modebuf[4];
	#ifdef DEBUG
	printf("### updateResolution():  bit=%d\n", vbits );
	#endif
	// 時計の針の描画色をGUI色で再設定
	ndlCol[0] = MG_colorChange( 10 );	// 秒針
	ndlCol[1] = MG_colorChange( 8 );	// 長針
	ndlCol[2] = MG_colorChange( 8 );	// 短針
	clsColor = MG_colorChange( 4 );
	// EGB 仮想画面再初期化
	EGB_resolutionRam( GV_egbWork, 0x80, vbits, 32, 32, GV_workBuf );
	EGB_writePage( GV_egbWork, 0x80 ) ;
	EGB_color( GV_egbWork, 0, clsColor ) ;
	EGB_color( GV_egbWork, 1, clsColor ) ;
	EGB_writeMode( GV_egbWork, PSET ) ;
	EGB_paintMode( GV_egbWork, 0x02 ) ;
	EGB_pen( GV_egbWork, 0 ) ;
	EGB_writePage( GV_egbWork, 0x00 ) ;
	
	return NOERR;
}

/*===================================================================*/
/*  画面情報取得＆更新再開                                           */
/*===================================================================*/
int		startUpdate()
{
	updateFlag = NOERR;
	
	updateResolution();
	userIdleTask();

	return NOERR;
}

/*===================================================================*/
/*  ウィンドゥ消去                                                   */
/*===================================================================*/
int		eraseWin()
{
	MMI_SendMessage( ID_ICON[0], MM_SLEEP, 0 );
	MMI_SendMessage( ID_ICON[0], MM_ERASE, 0 );
	MMI_SendMessage( ID_ICON[0], MM_DETACH, 0 );
	return NOERR;
}

/*===================================================================*/
/*  アイコン化されたウィンドゥの関数                                 */
/*===================================================================*/
int		func_ICON( int kobj, int messId, int ac, EVENT *pev, int trigger )
{
	HYPER	hyp;
	FRAME	fr;

	if( messId == MM_SHOW ){
		userIdleTask();
		return ERROR;
	}
	// ※ ここでダブルクリックなら関数起動 / それ以外ならMOVE
	if( messId == MM_MOUSEON ){
		// 枠座標を得る
		MMI_SendMessage( kobj, MM_GETHYPER, 1, &hyp );
		fr = hyp.fr;
		do {
			MMI_iosense();
		} while( (MMI_GetEvnt((EVMOSUP|EVMOSDRAG), &pev)) < NOERR);
		if ( pev->what == EVMOSUP ){
			// ダブルクリックチェック
			if ( MMI_DoubleClickCheck(&fr, pev) != FALSE ){
				/*========================================================*/
				/* このサンプルではダブルクリックで終了していますが、     */
				/* 「アイコン→ウィンドゥ表示」の処理の方が望ましいかな?  */
				/*========================================================*/
				SaveResumeFile();			/* リジューム情報をセーブ     */
				MMI_SetHaltFlag( TRUE );	/* イベントループを終了させる */
			}
			return NOERR;
		}
		// ダブルクリックでは無いのでMOVE処理へ
	}
	// エラーで抜けるとWINDOW部品本来の動作をしまっする♪
	return ERROR;
}

