基于Android4.0版本截屏+node.js服务器接收显示

最近在开发同屏,做了个Dome,记录下成果!

首先是web端和服务器端,这里我使用node.js来做的

服务端:service.js

var http = require('http');
var sio = require('socket.io');
var fs = require('fs');
var server = http.createServer(function(req,res){
    res.writeHead(200,
        {
            'Content-type':'text/html'
        }
    );
    res.end(fs.readFileSync('./main.html'));
});

server.listen(8083);
var socket = sio.listen(server);
var webs = socket;
socket.on('connection',function(socket){
    console.log("Connection...");

	
	setTimeout(function(){
		socket.emit('news',{shuju:''});
	},1000);

	setTimeout(function(){
		socket.emit('news',{shuju:''});
	},3000);
	

	socket.on('users',function(data){
		console.log(data)
		if(data=="web"){
			webs = socket;
		}
	});

	socket.on('data',function(data){
		console.log("data is OK..");
		webs.emit('news',{shuju:data});
	});
   
})
web端接收图片显示:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>MAIN</title>
   
</head>
<body>
    <img id="imgs" style="width:100%" src=""/>
</body>
<script src="http://abcwhatsappad.com:8080/wa/dist/easyui/jquery.min.js"></script> 
 <script src="http://cdn.bootcss.com/socket.io/1.7.3/socket.io.js"></script>
    <script>
        var socket = io.connect("ws://192.168.1.194:8083");
		socket.emit("users","web");
        socket.on('news',function(data){
			$(function(){
				$("#imgs").attr("src",data.shuju);
			});
        });
    </script>
</html>

接下来是Android端,这里要求是在5.0以下版本上开发同屏功能,我找了好多网上案例,现在也只能做到截图传输,效果也不尽人意。

首先导入io.socket有关的jar包:engine.io-client-0.8.3.jar, okhttp-3.6.0.jar,okio-1.11.0.jar,socket.io-client-0.8.4-SNAPSHOT.jar

连接服务器:SocketStreamWork.java

public class SocketStreamWork implements Runnable {
	private static final String TAG = "Alin-SocketStreamWork";
	private final static String SOCKET_SERVER_URL = "http://192.168.1.194:8083";
	private static final String IMAGE_TITLE = "data:image/png;base64,";

	private Socket mSocket;
	private boolean isConnected;
	private String result;

	public SocketStreamWork(String result) {
		this.result = result;
	}

	@Override
	public void run() {
		//建立Socket连接
		Log.e(TAG, "run");
		initSocketHttp();
		connectSocket();
	}
	
	private void initSocketHttp() {
		try {
			mSocket = IO.socket( SOCKET_SERVER_URL ); // 初始化Socket
			Log.e(TAG, "initSocketHttp");
		} catch ( URISyntaxException e ) {
			e.printStackTrace();
		}
	}

	private void connectSocket() {
		try {
			mSocket.connect();
			mSocket.emit("data", IMAGE_TITLE + result);
			Log.e(TAG, "connectSocket");	
		} catch (Exception e ) {
			e.printStackTrace();
		}

		mSocket.on( Socket.EVENT_CONNECT, onConnect );// 连接成功
		mSocket.on( Socket.EVENT_DISCONNECT, onDisconnect );// 断开连接
		mSocket.on( Socket.EVENT_CONNECT_ERROR, onConnectError );// 连接异常
		mSocket.on( Socket.EVENT_CONNECT_TIMEOUT, onConnectTimeoutError );// 连接超时
		mSocket.on( "data", onConnectMsg );// 监听消息事件回调
	}

	private void disConnectSocket() {
		mSocket.disconnect();
		mSocket.off( Socket.EVENT_CONNECT, onConnect );// 连接成功
		mSocket.off( Socket.EVENT_DISCONNECT, onDisconnect );// 断开连接
		mSocket.off( Socket.EVENT_CONNECT_ERROR, onConnectError );// 连接异常
		mSocket.off( Socket.EVENT_CONNECT_TIMEOUT, onConnectTimeoutError );// 连接超时
		mSocket.off( "data", onConnectMsg );// 监听消息事件回调
	}

	private Emitter.Listener onConnectMsg = new Emitter.Listener() {// 监听消息事件回调
		@Override
		public void call( final Object... args ) {
			// 在这里处理你的消息
			Log.e( TAG, "服务器返回来的消息 : " + args[0] );
		}
	};

	/*实现消息回调接口*/
	private Emitter.Listener onConnect = new Emitter.Listener() {// 连接成功
		@Override
		public void call( final Object... args ) {
			Log.e( TAG, "连接成功 ");
			if (!isConnected) { // 如果已经断开,重新发送
				/*try {
					mSocket.emit("data", result);
				} catch (Exception e ) {
					e.printStackTrace();
				}*/
				isConnected = true;
			}
		}
	};

	private Emitter.Listener onDisconnect = new Emitter.Listener() {// 断开连接
		@Override
		public void call( Object... args ) {
			Log.e( TAG, "断开连接 " + args[0] );
			isConnected = false;
		}
	};

	private Emitter.Listener onConnectError = new Emitter.Listener() {// 连接异常
		@Override
		public void call( final Object... args ) {
			Log.e( TAG, "连接 失败" + args[0] );
		}
	};

	private Emitter.Listener onConnectTimeoutError = new Emitter.Listener() {// 连接超时
		@Override
		public void call( final Object... args ) {
			Log.e( TAG, "连接 超时" + args[0] );

		}
	};
}
截取屏幕图片:ScreenUtil.java 

这里使用的是adb截屏功能,截取的图片是保存在路径里的,要获取还是要用FileInputStream()方法获取

public class ScreentUtil {
	private static final String TAG = "ScreentShotUtil";
	private static final String CLASS1_NAME = "android.view.SurfaceControl";
	private static final String CLASS2_NAME = "android.view.Surface";
	private static final String METHOD_NAME = "screenshot";

	private static ScreentUtil screentShotUtil;
	private Display mDisplay;
	private DisplayMetrics mDisplayMetrics;
	private Matrix mDisplayMatrix;
	private WindowManager wm;
	private SimpleDateFormat format;
	private static File outputImage = null;

	public static ScreentUtil getInstance(){
		synchronized (ScreentUtil.class) {
			if (screentShotUtil == null){
				screentShotUtil = new ScreentUtil();
			}
		}
		return screentShotUtil;
	}

	private Bitmap screenShot(int width, int height){

		Log.i(TAG, "android.os.Build.VERSION.SDK : " + android.os.Build.VERSION.SDK_INT);
		Class<?> surfaceClass = null;
		Method method = null;
		try{
			Log.i(TAG, "width : " + width);
			Log.i(TAG, "height : " + height);
			if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2){
				surfaceClass = Class.forName(CLASS1_NAME);
			} else {
				surfaceClass = Class.forName(CLASS2_NAME);
			}

			method = surfaceClass.getDeclaredMethod(METHOD_NAME, new Class[]{int.class, int.class});
			method.setAccessible(true);
			return (Bitmap) method.invoke(null, width, height);
		}catch (NoSuchMethodException e){
			Log.e(TAG, e.toString());
		}catch (IllegalArgumentException e){
			Log.e(TAG, e.toString());
		}catch (IllegalAccessException e){
			Log.e(TAG, e.toString());
		}catch (InvocationTargetException e){
			Log.e(TAG, e.toString());
		}catch (ClassNotFoundException e){
			Log.e(TAG, e.toString());
		}
		return null;
	}

	/**
	 * Takes a screenshot of the current display and shows an animation.
	 * @return 
	 */
	@SuppressLint("NewApi")
	public Bitmap ScreenShot(Context context){
		Bitmap mBitmap = null;
		if(outputImage == null){
			format = new SimpleDateFormat("yyyyMMddHHmmss");
			String fileName = format.format(new Date(System.currentTimeMillis())) + ".png";
            File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM); 
            outputImage = new File(path, fileName);  
			Log.e("Alin", "path : " + path + " , outputImage : " + outputImage);
		}
		if(ShellUtil.checkRootPermission()){
			if(android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH){
				ShellUtil.execCommand("/system/bin/screencap -p "+ outputImage, true);
				Log.d("Alin", "保存截图");
				if(outputImage.exists()){
					try {
						mBitmap = FileUtil.getInstance().getBitmapFromBytes(FileUtil.getInstance().readInputStream(new FileInputStream(outputImage)), null);
					} catch (Exception e) {
						e.printStackTrace();
					}
				}
			}
		} else {
			if(android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH){
				wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
				mDisplay = wm.getDefaultDisplay();
				mDisplayMatrix = new Matrix();
				mDisplayMetrics = new DisplayMetrics();
				// We need to orient the screenshot correctly (and the Surface api seems to take screenshots
				// only in the natural orientation of the device :!)
				mDisplay.getRealMetrics(mDisplayMetrics);
				float[] dims = {
						mDisplayMetrics.widthPixels/2, mDisplayMetrics.heightPixels/2
				};
				float degrees = getDegreesForRotation(mDisplay.getRotation());
				boolean requiresRotation = (degrees > 0);
				if (requiresRotation){
					// Get the dimensions of the device in its native orientation
					mDisplayMatrix.reset();
					mDisplayMatrix.preRotate(-degrees);
					mDisplayMatrix.mapPoints(dims);
					dims[0] = Math.abs(dims[0]);
					dims[1] = Math.abs(dims[1]);
				}
				Bitmap mScreenBitmap = screenShot((int) dims[0], (int) dims[1]);
				if (requiresRotation){
					// Rotate the screenshot to the current orientation
					Bitmap ss = Bitmap.createBitmap(mDisplayMetrics.widthPixels/2, mDisplayMetrics.heightPixels/2,
							Bitmap.Config.RGB_565);
					Canvas c = new Canvas(ss);
					c.translate(ss.getWidth() / 2, ss.getHeight() / 2);
					c.rotate(degrees);
					c.translate(-dims[0] / 2, -dims[1] / 2);
					c.drawBitmap(mScreenBitmap, 0, 0, null);
					c.setBitmap(null);
					mScreenBitmap = ss;
					if (ss != null && !ss.isRecycled()){
						ss.recycle();
					}
				}
				// If we couldn't take the screenshot, notify the user
				if (mScreenBitmap == null){
					Toast.makeText(context, "Alin :  screen shot fail", Toast.LENGTH_SHORT).show();
				}else {
					// Optimizations
					mScreenBitmap.setHasAlpha(false);
					mScreenBitmap.prepareToDraw();
					saveBitmap2file(context, mScreenBitmap, outputImage.toString());
				}
				mBitmap = mScreenBitmap;
			}
		}
		return mBitmap;
	}

	public void saveBitmap2file(Context context, Bitmap bmp, String fileName){
		//Log.e("Alin", bmp.getWidth()+"----------"+fileName);
		int quality = 100;
		ByteArrayOutputStream baos = new ByteArrayOutputStream();
		bmp.compress(Bitmap.CompressFormat.JPEG, quality, baos);
		InputStream is = new ByteArrayInputStream(baos.toByteArray());
		byte[] buffer = new byte[1024];
		int len = 0;
		File file = new File(fileName);
		if (!file.exists()){
			try{
				file.getParentFile().mkdir();
				file.getParentFile().createNewFile();
			}catch (IOException e){
				Log.e("Alin", e.toString());
			}
		}else{
			try{
				file.getParentFile().delete();
				file.getParentFile().createNewFile();
			}catch (IOException e){
				Log.e("Alin", e.toString());
			}
		}
		FileOutputStream stream = null;
		try{
			stream = new FileOutputStream(file);
			while ((len = is.read(buffer)) != -1){
				stream.write(buffer, 0, len);
			}
			stream.flush();
		}catch (FileNotFoundException e){
			Log.i(TAG, e.toString());
		}catch (IOException e){
			Log.i(TAG, e.toString());
		}finally{
			if (is != null) {
				try{
					is.close();
				} catch (IOException e) {
					Log.i(TAG, e.toString());
				}
			}
			if (stream != null){
				try{
					stream.close();
				}catch (IOException e){
					Log.i(TAG, e.toString());
				}
			}
		}
		if (bmp != null && !bmp.isRecycled()){
			bmp.recycle();
		}
	}

	/**
	 * @return the current display rotation in degrees
	 */
	private float getDegreesForRotation(int value){
		switch (value){
		case Surface.ROTATION_90:
			return 360f - 90f;
		case Surface.ROTATION_180:
			return 360f - 180f;
		case Surface.ROTATION_270:
			return 360f - 270f;
		}
		return 0f;
	}
}
adb shell命令文件ShellUtil.java

/**
 * ShellUtils
 * <ul>
 * <strong>Check root</strong>
 * <li>{@link ShellUtil#checkRootPermission()}</li>
 * </ul>
 * <ul>
 * <strong>Execte command</strong>
 * <li>{@link ShellUtil#execCommand(String, boolean)}</li>
 * <li>{@link ShellUtil#execCommand(String, boolean, boolean)}</li>
 * <li>{@link ShellUtil#execCommand(List, boolean)}</li>
 * <li>{@link ShellUtil#execCommand(List, boolean, boolean)}</li>
 * <li>{@link ShellUtil#execCommand(String[], boolean)}</li>
 * <li>{@link ShellUtil#execCommand(String[], boolean, boolean)}</li>
 * </ul>
 */
public class ShellUtil {

    public static final String COMMAND_SU       = "su";
    public static final String COMMAND_SH       = "sh";
    public static final String COMMAND_EXIT     = "exit\n";
    public static final String COMMAND_LINE_END = "\n";

    private ShellUtil() {
        throw new AssertionError();
    }

    /**
     * check whether has root permission
     * 
     * @return
     */
    public static boolean checkRootPermission() {
        return execCommand("echo root", true, false).result == 0;
    }

    /**
     * execute shell command, default return result msg
     * 
     * @param command command
     * @param isRoot whether need to run with root
     * @return
     * @see ShellUtil#execCommand(String[], boolean, boolean)
     */
    public static CommandResult execCommand(String command, boolean isRoot) {
        return execCommand(new String[] {command}, isRoot, true);
    }

    /**
     * execute shell commands, default return result msg
     * 
     * @param commands command list
     * @param isRoot whether need to run with root
     * @return
     * @see ShellUtil#execCommand(String[], boolean, boolean)
     */
    public static CommandResult execCommand(List<String> commands, boolean isRoot) {
        return execCommand(commands == null ? null : commands.toArray(new String[] {}), isRoot, true);
    }

    /**
     * execute shell commands, default return result msg
     * 
     * @param commands command array
     * @param isRoot whether need to run with root
     * @return
     * @see ShellUtil#execCommand(String[], boolean, boolean)
     */
    public static CommandResult execCommand(String[] commands, boolean isRoot) {
        return execCommand(commands, isRoot, true);
    }

    /**
     * execute shell command
     * 
     * @param command command
     * @param isRoot whether need to run with root
     * @param isNeedResultMsg whether need result msg
     * @return
     * @see ShellUtil#execCommand(String[], boolean, boolean)
     */
    public static CommandResult execCommand(String command, boolean isRoot, boolean isNeedResultMsg) {
        return execCommand(new String[] {command}, isRoot, isNeedResultMsg);
    }

    /**
     * execute shell commands
     * 
     * @param commands command list
     * @param isRoot whether need to run with root
     * @param isNeedResultMsg whether need result msg
     * @return
     * @see ShellUtil#execCommand(String[], boolean, boolean)
     */
    public static CommandResult execCommand(List<String> commands, boolean isRoot, boolean isNeedResultMsg) {
        return execCommand(commands == null ? null : commands.toArray(new String[] {}), isRoot, isNeedResultMsg);
    }

    /**
     * execute shell commands
     * 
     * @param commands command array
     * @param isRoot whether need to run with root
     * @param isNeedResultMsg whether need result msg
     * @return <ul>
     *         <li>if isNeedResultMsg is false, {@link CommandResult#successMsg} is null and
     *         {@link CommandResult#errorMsg} is null.</li>
     *         <li>if {@link CommandResult#result} is -1, there maybe some excepiton.</li>
     *         </ul>
     */
    public static CommandResult execCommand(String[] commands, boolean isRoot, boolean isNeedResultMsg) {
        int result = -1;
        if (commands == null || commands.length == 0) {
            return new CommandResult(result, null, null);
        }

        Process process = null;
        BufferedReader successResult = null;
        BufferedReader errorResult = null;
        StringBuilder successMsg = null;
        StringBuilder errorMsg = null;

        DataOutputStream os = null;
        try {
            process = Runtime.getRuntime().exec(isRoot ? COMMAND_SU : COMMAND_SH);
            os = new DataOutputStream(process.getOutputStream());
            for (String command : commands) {
                if (command == null) {
                    continue;
                }

                // donnot use os.writeBytes(commmand), avoid chinese charset error
                os.write(command.getBytes());
                os.writeBytes(COMMAND_LINE_END);
                os.flush();
            }
            os.writeBytes(COMMAND_EXIT);
            os.flush();

            result = process.waitFor();
            // get command result
            if (isNeedResultMsg) {
                successMsg = new StringBuilder();
                errorMsg = new StringBuilder();
                successResult = new BufferedReader(new InputStreamReader(process.getInputStream()));
                errorResult = new BufferedReader(new InputStreamReader(process.getErrorStream()));
                String s;
                while ((s = successResult.readLine()) != null) {
                    successMsg.append(s);
                }
                while ((s = errorResult.readLine()) != null) {
                    errorMsg.append(s);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (os != null) {
                    os.close();
                }
                if (successResult != null) {
                    successResult.close();
                }
                if (errorResult != null) {
                    errorResult.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }

            if (process != null) {
                process.destroy();
            }
        }
        return new CommandResult(result, successMsg == null ? null : successMsg.toString(), errorMsg == null ? null
                : errorMsg.toString());
    }

    /**
     * result of command
     * <ul>
     * <li>{@link CommandResult#result} means result of command, 0 means normal, else means error, same to excute in
     * linux shell</li>
     * <li>{@link CommandResult#successMsg} means success message of command result</li>
     * <li>{@link CommandResult#errorMsg} means error message of command result</li>
     * </ul>
     * 
     * @author <a href="http://www.trinea.cn" target="_blank">Trinea</a> 2013-5-16
     */
    public static class CommandResult {

        /** result of command **/
        public int    result;
        /** success message of command result **/
        public String successMsg;
        /** error message of command result **/
        public String errorMsg;

        public CommandResult(int result) {
            this.result = result;
        }

        public CommandResult(int result, String successMsg, String errorMsg) {
            this.result = result;
            this.successMsg = successMsg;
            this.errorMsg = errorMsg;
        }
    }
}
  文件操作类FileUtil.java 

public class FileUtil {
	
	private static FileUtil mFileUtil;
	
	public static FileUtil getInstance(){
		synchronized (FileUtil.class) {
			if (mFileUtil == null){
				mFileUtil = new FileUtil();
			}
		}
		return mFileUtil;
	}
    
    /**
     * InputStream to byte
     * @param inStream
     * @return
     * @throws Exception
     */
    public byte[] readInputStream(InputStream inStream) throws Exception { 
        byte[] buffer = new byte[1024]; 
        int len = -1; 
        ByteArrayOutputStream outStream = new ByteArrayOutputStream();
        
        while ((len = inStream.read(buffer)) != -1) { 
            outStream.write(buffer, 0, len); 
        }
        
        byte[] data = outStream.toByteArray(); 
        outStream.close(); 
        inStream.close();
        
        return data; 
   } 
    
   /**
    * Byte to bitmap
    * @param bytes
    * @param opts
    * @return
    */
   public Bitmap getBitmapFromBytes(byte[] bytes, BitmapFactory.Options opts) {
       if (bytes != null){
           if (opts != null){ 
               return BitmapFactory.decodeByteArray(bytes, 0, bytes.length,opts); 
           }
           else{
               return BitmapFactory.decodeByteArray(bytes, 0, bytes.length); 
           }
       }
       
       return null; 
   } 
}
使用Base64加密传输图片BitmapUtil.java
public class BitmapUtil {
	
	private static final String TAG = "Alin";
	
	private static BitmapUtil mBitmapUtil;
	/** 
	 * bitmap转为base64 
	 * @param bitmap 
	 * @return 
	 */  
	
	public static BitmapUtil getInstance(){
		synchronized (BitmapUtil.class) {
			if (mBitmapUtil == null){
				mBitmapUtil = new BitmapUtil();
			}
		}
		return mBitmapUtil;
	}
	
	public String bitmapToBase64(Bitmap bitmap) {  
		Log.d(TAG, "-----bitmapToBase64-----" + bitmap);
		String result = null;  
		ByteArrayOutputStream baos = null;  
		try {  
			if (bitmap != null) {  
				baos = new ByteArrayOutputStream();  
				bitmap.compress(Bitmap.CompressFormat.JPEG, 40, baos);  

				baos.flush();  
				baos.close();  

				byte[] bitmapBytes = baos.toByteArray();  
				result = Base64.encodeToString(bitmapBytes, Base64.DEFAULT);  
			}  
		} catch (IOException e) {  
			e.printStackTrace();  
		} finally {  
			try {  
				if (baos != null) {  
					baos.flush();  
					baos.close();  
				}  
			} catch (IOException e) {  
				e.printStackTrace();  
			}  
		}  
		return result;  
	}  

	/** 
	 * base64转为bitmap 
	 * @param base64Data 
	 * @return 
	 */  
	public Bitmap base64ToBitmap(String base64Data) {  
		Log.d(TAG, "-----base64ToBitmap-----" + (base64Data == null));
		byte[] bytes = Base64.decode(base64Data, Base64.DEFAULT);  
		return BitmapFactory.decodeByteArray(bytes, 0, bytes.length);  
	}  
}
最后我是用 service服务来运行的,这里就不在贴出来了。

猜你喜欢

转载自blog.csdn.net/alin693/article/details/78054572