/*<Header>==============================================================
*
*	俺を見てくれ〜 / "MAIN.C"
*
*		[ EIN(tm) project : サンプルプログラム ]
*
*	COPYRIGHT  Nam  1995, All rights reserved.
*
*	配付・組込・改変・商利用すべて自由。ただし無保証っす
*
*-----------------------------------------------------------------------
*	V1.0L01α	95.08.09/Nam	プロトタイプ
*	V1.0L01β	95.08.14/Nam	猫の手スクロール
*	V1.0L01γ	95.08.19/Nam	画面行数より短い時のスクロールバー化け回避
*	V1.0L02α	95.10.31/Nam	存在しないファイル名のチェック/DRAG&DROP対応
*</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	<pocket.h>			// このサンプルでのみ必要

#include	"..\ein\eintm.h"	// EIN(TM)関連ライブラリのヘッダ
#include	"..\eindd\eindd.h"	// DRAG&DROPライブラリのヘッダ


#define	ERROR	(-1)
#define	PSET	(0)

#define	DISPMODE_ICON	(0)
#define	DISPMODE_PANEL	(1)

#define	ID_text			(6)
#define	ID_scr			(9)
#define	ID_fname		(8)

int		ID_ICON[1];					// アイコン部品のID
int		ID_IWIN[11];				// メモウィンドゥ部品のID
int		ID_IWINMN0[12];				//    〃  メニュー部品のID
int		ID_ABOUT[10];
int		ID_JUMP[6];

int		dispMode=DISPMODE_PANEL;	// 表示状態(0:アイコン,1:メモウィンドゥ)

char	fname[128]="";				// ファイル名保持用
char	fstr[16]="";				// ファイル名表示用

/*= リジューム用設定=================================================*/
FRAME	RSM_ICONFR;
FRAME	RSM_IWINFR;
char	RSMID[]="%%% BROS.EXG %%%";
char	RSMTITLE[]="俺を見てくれ〜  V1.1 L12A";
char	RSMFNAME[128];
int		RSMDMODE=DISPMODE_ICON;
/*===================================================================*/

void	( *GV_defIdleTask )() ;

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( int argc, char **argv )
{
	extern int	APL_init() ;
	extern int	func_HIDE();
	extern int	func_show();
	extern int	func_load();
	extern int	func_activateText();
	extern void	func_updateScroll();

	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 ){
				// 起動パラメータチェック
				if( argc > 1 ){
					strncpy( fname, argv[1], 80 );
					func_load();
					func_show();
					func_activateText();
					func_updateScroll();
				}
				// メインループ
				MMI_ExecSystem();
			}
			// アイドルの解除
//			resetIdleTask();
			// ウィンドゥ消去(※SHELLに消去させると遅いので)
			func_HIDE( ID_ICON[0] );
			func_HIDE( ID_IWIN[0] );
		}
	}
	// 終了処理
	MMI_Close() ;
}

/*===================================================================*/
/*  初期化処理                                                       */
/*===================================================================*/
int APL_init()
{
	extern	MMIINIT	initDataRES_ICON ;
	extern	MMIINIT	initDataRES_IWIN ;
	extern	MMIINIT	initDataRES_ABUT ;
	extern	MMIINIT	initDataRES_JUMP ;

	extern	int ICONmaxId;
	extern	unsigned char *ICONTbl[];

	extern int	userFunc();

	HYPER	hyp;
	register int	ret ;

	// GUI部品の初期化
	if ((ret = MMI_initHyper()) < 0)			//	ハイパー型
		return ret;
	if ((ret = MMI_initDialogL40()) < 0)		//	ダイアログ型
		return ret;
	if ((ret = MMI_initAlertL40()) < 0)			//	アラート型
		return ret;
	if ((ret = MMI_initMessageL40()) < 0)		//	メッセージ型
		return ret;
	if ((ret = MMI_initMenuL40()) < 0)			//	メニュー型
		return ret;
	if ((ret = MMI_initButtonL40()) < 0)		//	ボタン型
		return ret;
	if ((ret = MMI_initDrawButtonL40()) < 0)	//	ドロウボタン型
		return ret;
//	if ((ret = MMI_initToggleIconL40()) < 0)	//	トグルボタン型
//		return ret;
	if ((ret = MMI_initIconL40()) < 0)			//	アイコン型
		return ret;
	if ((ret = MMI_initScrollBarL40()) < 0)		//	スクロールバー型
		return ret;
	if ((ret = MMI_initTextL40()) < 0)			//	テキスト型
		return ret;
	if ((ret = MMI_initMenuItemL40()) < 0)		//	メニューアイテム型
		return ret;
	if ((ret = MMI_initWindowL40()) < 0)		//	ウィンドウ型
		return ret;
	if ((ret = MMI_initNumBoxL40()) < 0)		//	数値入力パネル型
		return ret;
	/*	ファイルダイアログを使うときはさらにリストメニューの初期化も必要	*/
	/*	リストメニュー型部品の初期化			*/
	if ((ret = MMI_initListMenuL40()) < 0)
		return ret ;

	// リソースを登録
	if ((ret = MMI_Init(&initDataRES_ICON)) < 0)
		return ret ;
	if ((ret = MMI_Init(&initDataRES_IWIN)) < 0)
		return ret ;
	if ((ret = MMI_Init(&initDataRES_JUMP)) < 0)
		return ret ;
	if ((ret = MMI_Init(&initDataRES_ABUT)) < 0)
		return ret ;

	// ユーザアイコンを登録
	MMI_SetIconTable( ICONTbl, ICONmaxId ) ;

	// GUI色をメニュー色に設定
	EIN_initGuiColor();

	// テキストエディタライブラリを初期化
	TE_Init();

	/*----------------------------------------------------------------*/
	/*  リジューム読み込み (※のついた処理は必ず行ってください)       */
	/*----------------------------------------------------------------*/
	char	work[RSMWORKSIZE];	// リジュームマネージャのワークエリア
	char	buf[1024];
	char	*ptr;
	int		size, x, y;
	char	s[128];

	// ※リジュームマネージャの初期化(アプリの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 ) ){
			if ( x<-31 ){	x=0;	}
			if ( y<-31 ){	y=0;	}
			// 枠座標を得る
			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
			MMI_SendMessage( ID_ICON[0], MM_MOVE, 1, &RSM_ICONFR ) ;
		}
		// メモウィンドゥ表示位置
		if ( ((ptr = strstr(buf,"\nIWIN: ")) != NULL ) &&
			 (sscanf(ptr,"\nIWIN: %d %d\n", &x, &y) > 1 ) ){
			if ( x<-31 ){	x=0;	}
			if ( y<-31 ){	y=0;	}
			// 枠座標を得る
			MMI_SendMessage( ID_IWIN[0], MM_GETHYPER, 1, &hyp );
			RSM_IWINFR.lupx = x;
			RSM_IWINFR.lupy = y;
			RSM_IWINFR.rdwx = x+(hyp.fr.rdwx-hyp.fr.lupx);
			RSM_IWINFR.rdwy = y+(hyp.fr.rdwy-hyp.fr.lupy);
			#ifdef DEBUG
			printf("IWIN: move to (%d,%d)\n", x, y);
			#endif
			MMI_SendMessage( ID_IWIN[0], MM_MOVE, 1, &RSM_IWINFR ) ;
		}
		// 表示状態
		if ( ((ptr = strstr(buf,"\nMODE: ")) != NULL ) &&
			 (sscanf(ptr,"\nMODE: %d\n", &x) > 0 ) ){
			dispMode = (x==0) ?DISPMODE_ICON :DISPMODE_PANEL;
			RSMDMODE = dispMode;
			#ifdef DEBUG
			printf("MODE: %d\n", x);
			#endif
		}
		// ファイル名を得る
		if ( ((ptr = strstr(buf,"\nFNAME: ")) != NULL ) &&
			 (sscanf(ptr,"\nFNAME: %s\n", &s) > 0 ) ){
			#ifdef DEBUG
			printf("FNAME: %s\n", s);
			#endif
			strcpy( RSMFNAME, s );
			strcpy( fname, s );
		}
	}

	// ウィンドウ類をベース部品にATTACH
	if ( dispMode == DISPMODE_ICON ){
		MMI_SendMessage( ID_ICON[0], MM_ATTACH, 1, MMI_GetBaseObj() ) ;
	} else {
		MMI_SendMessage( ID_IWIN[0], MM_ATTACH, 1, MMI_GetBaseObj() ) ;
	}

	// スイッチャー対応関数の登録
	MMI_SendMessage( MMI_GetBaseObj(), MM_SETEXEC, 1, userFunc ) ;

	// タスクリストにタイトルを登録
	MMI_CallMessage( MMI_GetApliId(), GM_TITLE, (int)RSMTITLE, 0 ) ;

	//	背景を表示
	MMI_SendMessage( MMI_GetBaseObj(), MM_SHOW, 0 ) ;

	return NOERR ;
}

/*===================================================================*/
/* リジューム情報更新 (※がついた処理は必ず行って下さい)             */
/*===================================================================*/
void	SaveResumeFile()
{
	char	work[RSMWORKSIZE];
	char	buf[1024];
	char	aplpath[128];
	HYPER	hyp1, hyp2;
	TEHandle	*teh;
	register int ret;

	// テキストのサイズ取得
	MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &teh );

	// 枠座標を得る
	MMI_SendMessage( ID_ICON[0], MM_GETHYPER, 1, &hyp1 );
	MMI_SendMessage( ID_IWIN[0], MM_GETHYPER, 1, &hyp2 );
	
	// 移動していたらセーブ
	if ( (RSM_ICONFR.lupx  != hyp1.fr.lupx) ||
	     (RSM_ICONFR.lupy  != hyp1.fr.lupy) ||
	     (RSM_IWINFR.lupx  != hyp2.fr.lupx) ||
	     (RSM_IWINFR.lupy  != hyp2.fr.lupy) ||
		 (RSMDMODE         != dispMode)||		// 表示状態が違う
	     (strcmp(RSMFNAME,fname)!=0) ){			// ファイル名が違う
		// ※マネージャ初期化
		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_rsmBufPrintf( work, "ICON: %d %d", hyp1.fr.lupx, hyp1.fr.lupy );
		// メモウィンドゥ位置保存
		EIN_rsmBufPrintf( work, "IWIN: %d %d", hyp2.fr.lupx, hyp2.fr.lupy );
		// 表示モード保存
		EIN_rsmBufPrintf( work, "MODE: %d", dispMode );
		// メモファイル名保存
		EIN_rsmBufPrintf( work, "FNAME: %s", fname );
		// ※バッファ終端(^A)設定
		EIN_rsmBufTail( work );
		// ※バッファをセーブ
		ret = EIN_rsmSave( work );
		#ifdef DEBUG
		printf(" resume save size(%d)   ptr:(%d,%d)\n",ret, hyp1.fr.lupx, hyp1.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;
		RSM_IWINFR.lupx = hyp2.fr.lupx;	RSM_IWINFR.lupy = hyp2.fr.lupy;
		RSM_IWINFR.rdwx = hyp2.fr.rdwx;	RSM_IWINFR.rdwy = hyp2.fr.rdwy;
		RSMDMODE = dispMode;
	}
}

/*===================================================================*/
/*  イベントループ関数                                               */
/*===================================================================*/
int		userFunc( int apliId, int messId, int info, int data )
{
    // メタタスク拡張メッセージ ( GM_EXECUSER で、info で渡す )
    #define MTL_ICONIZE     0x8000      // アイコン化
    #define MTL_RESTORE     0x8001      // 復元
    #define MTL_MAXIMIZE    0x8002      // 最大化

	extern int	func_hide();
	extern int	func_show();
	extern int	func_load();
	extern int	func_activateText();
	extern void	func_updateScroll();

	register	int		ret ;

	ret = ILLEGAL_FUNCTION ;
	switch( messId ){
		//	他タスクからスイッチしてきた
		case GM_WAKE :
			break ;

		//	他タスクにスイッチする
		case GM_SLEEP :
			break ;

		//	画面変更前処理
		case GM_PRESCRCHG :
			break ;

		//	画面変更後処理
		case GM_POSTSCRCHG :
			break ;

		//	一時停止要求が出た
		case GM_PAUSE :
			break ;

		//	一時停止解除要求が出た
		case GM_CONTINUE :
			break ;

		//	終了要求が出た
		case GM_QUIT :
			// リジューム
			SaveResumeFile();
			//	正常終了
			MMI_SetHaltFlag( TRUE ) ;	//	イベントループを終了させるフラグ
			ret = NOERR ;
			break ;

		// メタタスク拡張メッセージ対応
		case GM_EXECUSER :				// メタタスクが送信する
			if ( info == MTL_ICONIZE ){
				// アイコン化メッセージだったら
				func_hide();
			} else if ( info == MTL_RESTORE ){
				// サイズ復元メッセージだったら
				func_show();
			} else if ( info == MTL_MAXIMIZE ){
				// 最大化メッセージだったら
				func_show();
			} else {
//				printf("%s\n", (char *)data);
				
				// その他なら、引数をファイル名としてオープン
				if ( (char *)data != NULL ){
					if ( strlen( (char *)data ) > 0 ){
						strncpy( fname, (char *)data, 80 );
						func_load();
						func_show();
						func_activateText();
						func_updateScroll();
					}
				}
			}
			ret = NOERR ;
			break;
		// DRAG&DROPメッセージ対応
		case GM_SENDDATA :
			// プールからファイル名を受け取る
			if ( EINDD_receivePool( info, data, fname, NULL )== NOERR ){
				func_load();
				func_show();
				func_activateText();
				func_updateScroll();
			}
			ret = NOERR ;
			break ;

	}

	return ret ;
}

/*===================================================================*/
/*  アイドルタスク関数(アイコン上に時計表示)                         */
/*===================================================================*/
void	userIdleTask()
{
}

/*===================================================================*/
/*  アイドルタスクに登録                                             */
/*===================================================================*/
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		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 ){
				// 「アイコン→ウィンドゥ表示」
				func_show();
			}
			return NOERR;
		}
		// ダブルクリックでは無いのでMOVE処理へ
	}
	// エラーで抜けるとWINDOW部品本来の動作をしまっする♪
	return ERROR;
}

/*===================================================================*/
/*  部品表示(汎用)                                                   */
/*===================================================================*/
int	func_SHOW( int id )
{
	char		clp[EINCLIPSIZE];
	register int	ret;

	// アイコンをATTACHする
	ret = MMI_SendMessage( id, MM_ATTACH, 1, MMI_GetBaseObj() ) ;
	// アイコンを表示する
	EIN_clipOpen( clp );
	MMI_SendMessage( id, MM_SHOW, 0 ) ;
	MMI_SendMessage( id, MM_WAKE, 0 ) ;
	EIN_clipClose( clp );

	return ret;
}

/*===================================================================*/
/*  部品消去(汎用)                                                   */
/*===================================================================*/
int	func_HIDE( int id )
{
	MMI_SendMessage( id, MM_SLEEP, 0 );
	MMI_SendMessage( id, MM_ERASE, 0 );
	MMI_SendMessage( id, MM_DETACH, 0 );
	return NOERR;
}

/*===================================================================*/
/*  ウィンドゥ表示 & アイコン消去                                    */
/*===================================================================*/
int	func_show()
{
	// 既にウィンドゥが表示されているなら何もしない
	if ( dispMode==DISPMODE_PANEL){
		return NOERR;
	}
	dispMode=DISPMODE_PANEL;

	// アイコンを消去する
	func_HIDE( ID_ICON[0] );
	// 入力ウィンドゥを表示する
	func_SHOW( ID_IWIN[0] );

	return NOERR;
}

/*===================================================================*/
/*  ウィンドゥ消去                                                   */
/*===================================================================*/
int	func_close()
{
	// アイコンを消去する
	func_HIDE( ID_ICON[0] );
	// ウィンドゥを消去する
	func_HIDE( ID_IWIN[0] );
	// リジューム
	SaveResumeFile();
	//	正常終了
	MMI_SetHaltFlag( TRUE ) ;	//	イベントループを終了させるフラグ

	return NOERR;
}

/*===================================================================*/
/*  ウィンドゥ一時消去                                               */
/*===================================================================*/
int	func_hide()
{
	// パネルが表示されていないなら何もしない
	if ( dispMode!=DISPMODE_PANEL){
		return ERROR;
	}
	dispMode=DISPMODE_ICON;

	// 入力ウィンドゥを消去する
	func_HIDE( ID_IWIN[0] );
	// アイコンを表示
	func_SHOW( ID_ICON[0] );

	return NOERR;
}

int	func_activateText()
{
	TEHandle	*th;
	
	// 状態取得
	MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &th );
	TE_Activate( th, 0, 0 );
	TE_dsplineJump( 0 );
	MG_mosDisp(2);
	TE_UpDate();
	MG_mosDisp(3);
	return NOERR;
}

int	func_load()
{
	#define	LBUFSIZE	(1024)
	char	buf[LBUFSIZE];
	int		csize;
	FILE	*fp;
	int		mptr;

	if ( (fp = EIN_fopen( fname, "r" )) == (FILE *)NULL ){
		return ERROR;
	}
	// マウスカーソルを時計に
	MG_PushPtr( MB_WAIT, &mptr );
	// テキストクリア
	MMI_SendMessage( ID_IWIN[ID_text], MM_SETTEXT, 3, NULL, 0, FALSE );
	// ファイル終端まで読み込み
	for ( ; (csize=fread(buf,1,LBUFSIZE,fp))>0;  ){
		MMI_SendMessage( ID_IWIN[ID_text], MM_SETTEXT, 3, buf, csize, TRUE );
	}
	fclose( fp );
	// マウスカーソル復帰
	MG_PopPtr( mptr );
	// ファイル名表示
	MMI_SendMessage( ID_IWIN[ID_fname], MM_SETMSG, 1, "             " );
	MMI_SendMessage( ID_IWIN[ID_fname], MM_SHOW, 0 );
	EIN_fnameNonDirectory( fname, fstr );
	MMI_SendMessage( ID_IWIN[ID_fname], MM_SETMSG, 1, fstr );
	MMI_SendMessage( ID_IWIN[ID_fname], MM_SHOW, 0 );

	return NOERR;
}

int	func_pocketOut()
{
	#define	RBUFSIZE	(1024)
	char	buf[RBUFSIZE];
	int		csize;
	int		mptr;

	if ( SCB_open()==NOERR ){
		if ( SCB_isFormatAvailable(SCBF_CSTR) > 0 ){
			// マウスカーソルを時計に
			MG_PushPtr( MB_WAIT, &mptr );
			// テキストクリア
			MMI_SendMessage( ID_IWIN[ID_text], MM_SETTEXT, 3, NULL, 0, FALSE );
			SCB_seek(SCBF_CSTR, 0, SCBS_SET);
			/* ファイル終端まで読み込み */
			for ( ; (csize=SCB_read(SCBF_CSTR, buf, RBUFSIZE))>0;  ){
				MMI_SendMessage( ID_IWIN[ID_text], MM_SETTEXT, 3, buf, csize, TRUE );
			}
//			MMI_SendMessage( ID_IWIN[ID_text], MM_SHOW, 0 );
			extern void	func_updateScroll();
			func_activateText();
			func_updateScroll();
			// マウスカーソル復帰
			MG_PopPtr( mptr );
		}
		SCB_close();
	}
	return NOERR;
}

int	func_pocketIn()
{
	#define	WBUFSIZE	(1024)
	char	buf[WBUFSIZE];
	int		csize, nsize, lfg, ret=NOERR;
	TEHandle		*teh;

	// テキストのサイズ取得
	MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &teh );
	csize = (int)(teh->Inchar);
	if ( csize < 1 ){
		#ifdef DEBUG
		printf("func_MWpocketIn(): No text.\n");
		#endif
		return NOERR;
	}
	if ( SCB_open()==NOERR ){
		SCB_empty();
		if ( csize > 0 ){
			/* テキスト終端まで書き込み */
			for ( lfg=FALSE, ret=NOERR; ; lfg=TRUE ){
				nsize = MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEXT, 3, buf, WBUFSIZE, lfg );
				if ( SCB_write(SCBF_CSTR,buf,((csize>WBUFSIZE)?WBUFSIZE:csize)) < 1 ){
					ret = ERROR;
					break;
				}
				if ( (csize = nsize) < 1 ){
					break;
				}
			}
		}
		strcpy(buf,"\n");
		SCB_write(SCBF_CSTR,buf,strlen(buf));
		SCB_seek(SCBF_CSTR, 0, SCBS_SET);
		SCB_close();
	}
	return ret;
}

int	func_new()
{
	MMI_SendMessage( ID_IWIN[ID_text], MM_SETTEXT, 3, NULL, 0, FALSE );
	MMI_SendMessage( ID_IWIN[ID_text], MM_SHOW, 0 );
	return NOERR;
}

int	func_fname()
{
	unsigned int	InfSw;
	char			*ExtStr[]={"*.*", NULL};
	unsigned int	MSlctCnt;
	int				Atr, i, ret;
	char			path[128], filename[12];
	FILE	*fp;

retry:
	ret = FDG_InitFileDlg();
	if ( ret < 0 ){
		return ERROR;
	}
	FDG_SaveCurDir();
//	// Qドライブはパスする
//	FDG_SetPassDrv("Q");
	FDG_SetTitle( "ファイル名", "実行", "取消" );
	InfSw=(FDG_TEXT|FDG_FILEONLY|FDG_REDRAW|FDG_ALERT);
	// 初期パスを設定
	// (指定ファイルの存在をチェック  '95.10.31)
	if ( (strlen(fname)>0) &&
		 (fp = EIN_fopen( fname, "r" )) != (FILE *)NULL ){
		fclose(fp);
		EIN_fnameNonDirectory( fname, filename );
		EIN_fnameDirectory( fname, path );
		// パス名末尾の"￥"を削除する(FDGの仕様)
		i=strlen(path)-1;
		if (i>0){
			path[i] = 0;
		}
		#ifdef DEBUG
		printf("%s\n%s\n", filename, path);
		#endif
		FDG_SetFileText( filename );
		ret=FDG_DspFileDlg( MMI_GetBaseObj(), InfSw, path, ExtStr, &MSlctCnt );
	} else {
		// ファイルが無い場合
		ret=FDG_DspFileDlg( MMI_GetBaseObj(), InfSw, NULL, ExtStr, &MSlctCnt );
	}
	FDG_RecovCurDir();
	// 取り消されたら終了
	if ( ret != 1 ){
		FDG_FreeFileDlg();
		return ERROR;
	}
	FDG_GetPathName( fname, &Atr, 0 );
	FDG_FreeFileDlg();
	// 小文字は大文字に
	for ( i=0; i<strlen(fname); i++ ){
		fname[i] = toupper( (int)fname[i] );
	}

	// ファイル読み込み
	if ( func_load()!=NOERR ){
		// 失敗しちゃったの
		strcpy(fname, "");
	}
	extern void	func_updateScroll();
	func_activateText();
	func_updateScroll();
	
	return NOERR;
}

void	func_updateScroll()
{
	int		textLen, line, prt, max, min, len, page, nprt, nmax;
	TEHandle	*th;

	// 状態取得
	MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &th );
	MMI_SendMessage( ID_IWIN[ID_scr],  MM_GETSCROLL, 5, &prt,&min,&max,&len,&page );
	TE_GetWindowLine( &len );			// 表示行数
	TE_GetTextLine2( th, &line );		// テキストの総行数
	// スクロールバーの値を再設定
	nmax = (line<len) ?(min + len -1) :(min + line - len -1);
	if (nmax<=len){ nmax=len-1; }
	// テキスト上の位置から
	TE_GetDspLine2( th, &nprt );
	nprt = (line > len) ?nprt+min :min;	// 画面より行数が少ないとき
	// スクロールバーを書換える
	nprt = (nprt>nmax) ?nmax :nprt;
	MMI_SendMessage( ID_IWIN[ID_scr],  MM_SETSCROLL, 5, nprt,min,nmax,len,page );
	MMI_SendMessage( ID_IWIN[ID_scr],  MM_SHOW, 0 );
	// テキスト再表示
	MMI_SendMessage( ID_IWIN[ID_text],  MM_EXEC, 2, 0, 0 );
}

int		func_scroll( int kobj, int messId, int ac, EVENT *pev, int trigger )
{
	int			oprt, prt, max, min, len, page;
	TEHandle	*th;
	
	// 状態取得
	MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &th );
	MMI_SendMessage( ID_IWIN[ID_scr],  MM_GETSCROLL, 5, &prt,&min,&max,&len,&page );
	TE_GetDspLine2( th, &oprt );		// テキストの表示開始行取得
	// スクロールバーの移動に従いテキストをスクロール
	MG_mosDisp(2);
	TE_Activate( th, 0, 0 );
	TE_Scroll( (((prt - min) >= oprt) ?1 :0), abs( prt - min - oprt ) );
	MG_mosDisp(3);

	return NOERR;
}

// ジャンプパネルを表示
int	func_jumpPanel()
{
	return func_SHOW( ID_JUMP[0] );
}

// ジャンプパネルを閉じて指定行へジャンプ
int	func_jump()
{
	extern void	func_updateScroll();

	int		var,min,max,delta,ptColum, line, dline;
	TEHandle	*th;

	MMI_SendMessage( ID_JUMP[3],  MM_GETNUMBOX, 5, &var,&min,&max,&delta,&ptColum );
	func_HIDE( ID_JUMP[0] );
	MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &th );
	TE_GetTextLine2( th, &line );		// テキスト行数
	TE_GetWindowLine( &dline );			// 表示行数
	dline = (dline<line) ?dline :0;
	var = (line>var) ?var :(line-dline);	// 文末を越えないように
	TE_Activate( th, 0, 0 );
	TE_dsplineJump( var );
	MG_mosDisp(2);
	TE_UpDate();
	MG_mosDisp(3);
	func_updateScroll();
	return NOERR;
}

// アバウト表示
int	func_about()
{
	return func_SHOW( ID_ABOUT[0] );
}

// アバウトクローズ
int	func_ABclose()
{
	func_HIDE( ID_ABOUT[0] );
	return NOERR;
}

/*===================================================================*/
/*  ※猫の手スクロール((c)MOP)                                       */
/*                                                                   */
/*  ・何の事はない、テキスト部品の上にボタンを一つ置いて、その関数を */
/*    拾ってマウスイベントを掴んでループするんである。               */
/*                                                                   */
/*===================================================================*/
int		func_drag( int kobj, int messId, int ac, EVENT *pev, int trigger )
{
	HYPER	hyp;
	FRAME	fr;
	int		textLen, line, prt, max, min, len, page, nprt, nmax, dstep;
	TEHandle	*th;
	int		btn, bx, by, ox, oy;
	int		sline, ostep=0, foo;
	int		mptr;

	// ※ ここでダブルクリックなら関数起動 / それ以外ならMOVE
	if( messId != MM_EXEC ){
		return ERROR;
	}
	if ( pev->what == EVMOSDN ){
		// マウスカーソルを掌に
		MG_PushPtr( MB_MOVE, &mptr );
		// スクロールバー書換えの為の準備
		MMI_SendMessage( ID_IWIN[ID_text], MM_GETTEHANDLE, 1, &th );
		MMI_SendMessage( ID_IWIN[ID_scr],  MM_GETSCROLL, 5, &prt,&min,&max,&len,&page );
		// 枠座標を得る
		MMI_SendMessage( kobj, MM_GETHYPER, 1, &hyp );
		fr = hyp.fr;
		// 一行あたりのドット数を算出
		dstep = th->txFont.sizeY + th->lineHeight;
		dstep = (dstep>0) ?dstep :13;
		// 現在のマウスカーソルの位置を読む
		MOS_rdpos( &btn, &ox, &oy );
		// ドラッグを止めるまでループ
		do {
			sline = 0;
			MOS_rdpos( &btn, &bx, &by );
			if( by != oy ){
				foo = (oy - by) / dstep;	// 何行分移動しているか?
				if ( foo != 0 ){
					sline = foo;
					ostep = sline;
					oy = by;
				}
			}
			if ( ((fr.lupy > by) || (by > fr.rdwy)) && (ostep != 0) ){
				sline = ostep;
			}
			if ( sline != 0 ){
				MG_mosDisp(2);
				TE_Scroll( ((sline>0) ?1 :0), abs(sline) );
				// スクロールバーを書換える
				TE_GetDspLine2( th, &nprt );
				nprt = (nprt>max) ?max :nprt;
				MMI_SendMessage( ID_IWIN[ID_scr],  MM_SETSCROLL, 5, nprt,min,max,len,page );
 				MMI_SendMessage( ID_IWIN[ID_scr],  MM_SHOW, 0 );
				MG_mosDisp(3);
			}
		} while( (btn & 0x01)!=0 );
		MG_PopPtr( mptr );
		return ERROR;
	}
	// エラーで抜けるとボタン型部品本来の動作をしまっする♪
	return ERROR;
}
