赞
踩
最近在开发同屏,做了个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服务来运行的,这里就不在贴出来了。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。