综合作业1:Android文本阅读器

   

    文本阅读器,区别于小说阅读器,不能读取大容量的文件。文件容量过大,就会导致读取的速度变得很慢。

    读取一个文本文件需要使用BufferedReader + FileReader来逐行读取内容。
还需要准备一个滚动面板配合TextView来显示内容。

如果文件比较大,读取时间会比较长,就需要使用ProgressDialog来建立进度条,提示用户当前正在加载数据。

 

首先,我们看下整体的实现效果:

 

注:这里设置只显示了SD卡中的文件夹以及文本文件,这是通过扩展名的判断实现的

 

实行步骤:第一步:

一.创建activity_main布局:

    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >


            android:id="@+id/title_text"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:gravity="center_vertical"
        android:text="当前位置: /mnt/sdcard"
        android:textSize="14sp" />


            android:id="@+id/list"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="8"
        android:cacheColorHint="#00000000" >
   


 

二、创建给列表添加数据的file_line.xml布局:两个TextView,一个是显示文件夹图片的,一个事显示文件夹名字的

    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal" >


            android:id="@+id/file_img"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="1" />


            android:id="@+id/file_name"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="4"
        android:textSize="14sp" />


 

三、创建显示文本的activity

 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:background="#ffffff" >
        android:id="@+id/detail"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:textColor="#000000"
        android:textSize="14sp" />

 

四、布局写完后,下面是老样子,先编写一个Globals的公共类。
用来动态计算手机的高度和宽度,等比放大缩小。还有初始化所有扩展名和图片的对应关系

import java.util.HashMap;
import java.util.Map;
import org.liky.txt.R;
import android.app.Activity;
public class Globals {
public static int SCREEN_WIDTH;
public static int SCREEN_HEIGHT;
// 建立一个Map集合, 里面封装了所有扩展名对应的图标图片, 以便进行文件图标的显示
public static Map allIconImgs = new HashMap();
public static void init(Activity a) {
SCREEN_WIDTH = a.getWindowManager().getDefaultDisplay().getWidth();
SCREEN_HEIGHT = a.getWindowManager().getDefaultDisplay().getHeight();
// 初始化所有扩展名和图片的对应关系
allIconImgs.put("txt", R.drawable.txt_file);
allIconImgs.put("mp3", R.drawable.mp3_file);
allIconImgs.put("mp4", R.drawable.mp4_file);
allIconImgs.put("bmp", R.drawable.image_file);
allIconImgs.put("gif", R.drawable.image_file);
allIconImgs.put("png", R.drawable.image_file);
allIconImgs.put("jpg", R.drawable.image_file);
allIconImgs.put("dir_open", R.drawable.open_dir);
allIconImgs.put("dir_close", R.drawable.close_dir);
}


}

五、创建一个自定义FileAdapter,用于读取,显示界面的列表中的数据,包括通过扩展名取得的图片和文件名:

import java.util.ArrayList;

import java.util.List;
import java.util.Map;
import org.liky.txt.R;
import org.liky.txt.util.Globals;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AbsListView.LayoutParams;
import android.widget.BaseAdapter;
import android.widget.TextView;
public class FileAdapter extends BaseAdapter {
private Context ctx;
private List> allValues = new ArrayList>();
public FileAdapter(Context ctx, List> allValues) {
this.ctx = ctx;
this.allValues = allValues;
}
@Override
public int getCount() {
return allValues.size();
}
@Override
public Object getItem(int arg0) {
return allValues.get(arg0);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = LayoutInflater.from(ctx).inflate(R.layout.file_line,
null);
// 设置高度
convertView.setLayoutParams(new LayoutParams(
LayoutParams.MATCH_PARENT, Globals.SCREEN_HEIGHT / 9));
}
// 取得组件
TextView fileImg = (TextView) convertView.findViewById(R.id.file_img);
fileImg.getLayoutParams().height = Globals.SCREEN_HEIGHT / 9;
TextView fileName = (TextView) convertView.findViewById(R.id.file_name);
// 取得数据,设置到组件里
Map map = allValues.get(position);
// 设置内容, 文字
fileName.setText(map.get("fileName").toString());
// 图片要根据扩展名取得
String extName = map.get("extName").toString();
// 取得图片的id
int imgId = Globals.allIconImgs.get(extName);
// 设置图片
fileImg.setBackgroundResource(imgId);
return convertView;
  }
}

 

六、创建mainActivity,显示的是activity_main.xml,这中包含一个点击事件,一个退出按钮的监听,还有对扩展名的截取,只显示文件夹和文本文件

package org.liky.txt;

 

import java.io.File;

import java.util.ArrayList;

import java.util.HashMap;

import java.util.List;

import java.util.Map;

 

import org.liky.txt.adapter.FileAdapter;

import org.liky.txt.util.Globals;

 

import android.app.Activity;

import android.app.AlertDialog.Builder;

import android.content.DialogInterface;

import android.content.DialogInterface.OnClickListener;

import android.content.Intent;

import android.os.Bundle;

import android.os.Environment;

import android.view.KeyEvent;

import android.view.View;

import android.widget.AdapterView;

import android.widget.AdapterView.OnItemClickListener;

import android.widget.AdapterView.OnItemLongClickListener;

import android.widget.ListView;

import android.widget.TextView;

 

public class MainActivity extends Activity {

 

private TextView titleText;

private ListView list;

 

private FileAdapter adapter;

private List> allValues = new ArrayList>();

 

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

 

Globals.init(this);

 

setContentView(R.layout.activity_main);

 

// 取得组件

titleText = (TextView) findViewById(R.id.title_text);

list = (ListView) findViewById(R.id.list);

 

// 准备数据

// 取得SD卡根目录

File root = Environment.getExternalStorageDirectory();

 

loadFileData(root);

 

// 建立Adapter

adapter = new FileAdapter(this, allValues);

 

list.setAdapter(adapter);

 

// 加入监听事件

list.setOnItemClickListener(new OnItemClickListener() {

@Override

public void onItemClick(AdapterView arg0, View arg1, int arg2,

long arg3) {

// 取得当前操作的数据

Map map = allValues.get(arg2);

// 判断所点的是文件还是文件夹

boolean dirFlag = (Boolean) map.get("dirFlag");

if (dirFlag) {

// 文件夹

// 建立该文件夹的File对象

// 取得绝对路径

String fullPath = (String) map.get("fullPath");

// 建立File

File dir = new File(fullPath);

 

// 先清空原有数据

allValues.clear();

 

if (!Environment.getExternalStorageDirectory()

.getAbsolutePath().equals(fullPath)) {

// 加入返回上一级的操作行

Map parent = new HashMap();

parent.put("fileName", "返回上一级");

parent.put("extName", "dir_open");

parent.put("dirFlag", true);

parent.put("fullPath", dir.getParent());

 

// 保存一个标志

parent.put("flag", "TRUE");

 

// 将这一行加入到数据集合中

allValues.add(parent);

}

 

// 加入新数据

loadFileData(dir);

 

// 使用Adapter通知界面ListView,数据已经被修改了,你也要一起改

adapter.notifyDataSetChanged();

} else {

// 弹出该文件的详细信息

File f = new File((String) map.get("fullPath"));

 

// 切换界面

Intent in = new Intent(MainActivity.this,

DetailActivity.class);

in.putExtra("fullPath", map.get("fullPath").toString());

startActivity(in);

 

}

 

}

});

 

list.setOnItemLongClickListener(new OnItemLongClickListener() {

@Override

public boolean onItemLongClick(AdapterView arg0, View arg1,

final int arg2, long arg3) {

// 取得数据

Map map = allValues.get(arg2);

final File f = new File(map.get("fullPath").toString());

 

if (f.isFile()) {

// 弹出确认框

Builder builder = new Builder(MainActivity.this);

builder.setTitle("提示");

builder.setMessage("确定要删除该文件(" + f.getName() + ")?");

builder.setPositiveButton("确定", new OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

// SD卡中的文件删除

if (f.exists()) {

f.delete();

}

 

// 将列表中的数据删除

allValues.remove(arg2);

// 通知

adapter.notifyDataSetChanged();

}

});

builder.setNegativeButton("取消", new OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

}

});

 

builder.create().show();

}

 

return false;

}

});

 

}

 

@Override

public boolean onKeyDown(int keyCode, KeyEvent event) {

// 根据keyCode判断用户按下了哪个键

if (keyCode == KeyEvent.KEYCODE_BACK) {

// 判断当前是否在SD卡跟目录.

// 取得第一行数据

Map map = allValues.get(0);

if ("TRUE".equals(map.get("flag"))) {

// 里面,需要返回上一级

list.performItemClick(list.getChildAt(0), 0, list.getChildAt(0)

.getId());

} else {

// 弹出提示框

Builder builder = new Builder(MainActivity.this);

builder.setTitle("提示");

builder.setMessage(",真的要离开我吗?");

builder.setPositiveButton("真的", new OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

// 关闭当前Activity

finish();

}

});

builder.setNegativeButton("再待会儿", new OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

 

}

});

builder.create().show();

}

 

return false;

}

 

return super.onKeyDown(keyCode, event);

}

 

private void loadFileData(File dir) {

// 列出该目录下的所有文件

File[] allFiles = dir.listFiles();

// 设置当前位置的提示信息

titleText.setText("当前位置: " + dir.getAbsolutePath());

 

// 判断

if (allFiles != null) {

// 循环

for (int i = 0; i < allFiles.length; i++) {

File f = allFiles[i];

Map map = new HashMap();

map.put("fileName", f.getName());

// 多保存一个文件的绝对路径,方便在进行点击时使用

map.put("fullPath", f.getAbsolutePath());

// 判断是文件夹还是文件

if (f.isDirectory()) {

// 是文件夹

map.put("extName", "dir_close");

map.put("dirFlag", true);

} else {

// 是文件

// 截取出扩展名

String extName = f.getName()

.substring(f.getName().lastIndexOf(".") + 1)

.toLowerCase();

map.put("extName", extName);

map.put("dirFlag", false);

}

 

// 只有文件夹或文本文件才要显示.

if (f.isDirectory() || "txt".equals(map.get("extName"))) {

allValues.add(map);

}

}

}

}

 

}

 

、建立一个DetailActivity的用以读取文本文件,和显示进度条。使用的是ProgressDialog来完成进度条的显示,但同时要加入线程操作。

需要建立一个Handler类,来处理消息接收的功能。

注:测试时提示错误,Android只允许主线程对UI界面显示的内容进行修改,子线程不允许修改界面。

但这里必须在子线程对界面进行修改,这种情况Android是通过消息通道机制来解决的。

package org.liky.txt;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStreamReader;
import android.app.Activity;
import android.app.ProgressDialog;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.TextView;
public class DetailActivity extends Activity {
private TextView detail;
// 声明这个Handler类
private Handler handler;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_detail);
detail = (TextView) findViewById(R.id.detail);
// 建立这个handler对象,并覆写handleMessage方法,该方法会在接收到子线程的消息时自动执行,进行界面修改
handler = new Handler() {
@Override
public void handleMessage(Message msg) {
// msg就是子线程发送过来的消息
// 修改text内容
detail.setText(msg.obj.toString());
}
};
// 建立进度条
final ProgressDialog dialog = new ProgressDialog(this);
// 设置为水平进度条
dialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
dialog.setTitle("提示");
dialog.setMessage("正在加载数据, 请稍候...");
// 设置最大值
File f = new File(getIntent().getStringExtra("fullPath"));
dialog.setMax((int) f.length());
dialog.show();
Thread t = new Thread() {
@Override
public void run() {
// 接收参数,并根据参数建立文件对象
File f = new File(getIntent().getStringExtra("fullPath"));
// 使用IO流读取
try {
BufferedReader reader = new BufferedReader(
new InputStreamReader(new FileInputStream(f), "GBK"));
String line = null;
StringBuilder builder = new StringBuilder();
// 定义保存读取内容大小的变量
int size = 0;
while ((line = reader.readLine()) != null) {
builder.append(line + "\r\n");
// 当读取了一部分内容后,根据读取内容的大小,然后增长进度
size += line.getBytes().length;
// 每增加超过10k的内容,就改变一次进度条
if (size >= 10240) {
dialog.incrementProgressBy(size);
size = 0;
// 睡眠一段时间
Thread.sleep(5);
}
}
reader.close();
// detail.setText(builder.toString());
Message msg = new Message();
// 将builder中的内容设置到msg里
msg.obj = builder;
// 发送消息
handler.sendMessage(msg);
// 关闭进度条
dialog.dismiss();
} catch (Exception e) {
e.printStackTrace();
      }
    }
  };
t.start();
  }
}


你可能感兴趣的:(综合作业1:Android文本阅读器)