From 93b5f5d20303aebd399c6c8a0aa419b2cc04c22b Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Fri, 14 Jun 2024 18:16:51 +0200 Subject: [PATCH 1/9] Add Screen Sharing sample --- Device-Screen-Sharing-Java/.gitignore | 15 ++ Device-Screen-Sharing-Java/README.md | 143 +++++++++++ Device-Screen-Sharing-Java/app/.gitignore | 1 + Device-Screen-Sharing-Java/app/build.gradle | 38 +++ .../app/src/main/AndroidManifest.xml | 29 +++ .../sample/screensharing/MainActivity.java | 240 ++++++++++++++++++ .../sample/screensharing/OpenTokConfig.java | 38 +++ .../screensharing/ScreenSharingCapturer.java | 170 +++++++++++++ .../screensharing/ScreenSharingManager.java | 65 +++++ .../screensharing/ScreenSharingService.java | 73 ++++++ .../app/src/main/res/layout/activity_main.xml | 34 +++ .../src/main/res/mipmap-hdpi/ic_launcher.png | Bin 0 -> 2614 bytes .../src/main/res/mipmap-mdpi/ic_launcher.png | Bin 0 -> 1442 bytes .../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin 0 -> 3519 bytes .../main/res/mipmap-xxhdpi/ic_launcher.png | Bin 0 -> 6153 bytes .../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin 0 -> 9013 bytes .../app/src/main/res/values-w820dp/dimens.xml | 6 + .../app/src/main/res/values/colors.xml | 6 + .../app/src/main/res/values/dimens.xml | 5 + .../app/src/main/res/values/strings.xml | 4 + .../app/src/main/res/values/styles.xml | 11 + Device-Screen-Sharing-Java/build.gradle | 9 + Device-Screen-Sharing-Java/gradle.properties | 20 ++ .../gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 53636 bytes .../gradle/wrapper/gradle-wrapper.properties | 6 + Device-Screen-Sharing-Java/gradlew | 160 ++++++++++++ Device-Screen-Sharing-Java/gradlew.bat | 90 +++++++ Device-Screen-Sharing-Java/settings.gradle | 16 ++ 28 files changed, 1179 insertions(+) create mode 100644 Device-Screen-Sharing-Java/.gitignore create mode 100644 Device-Screen-Sharing-Java/README.md create mode 100644 Device-Screen-Sharing-Java/app/.gitignore create mode 100644 Device-Screen-Sharing-Java/app/build.gradle create mode 100644 Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml create mode 100644 Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java create mode 100644 Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java create mode 100644 Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java create mode 100644 Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java create mode 100644 Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/values-w820dp/dimens.xml create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/values/colors.xml create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/values/dimens.xml create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml create mode 100644 Device-Screen-Sharing-Java/app/src/main/res/values/styles.xml create mode 100644 Device-Screen-Sharing-Java/build.gradle create mode 100644 Device-Screen-Sharing-Java/gradle.properties create mode 100644 Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar create mode 100644 Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties create mode 100755 Device-Screen-Sharing-Java/gradlew create mode 100644 Device-Screen-Sharing-Java/gradlew.bat create mode 100644 Device-Screen-Sharing-Java/settings.gradle diff --git a/Device-Screen-Sharing-Java/.gitignore b/Device-Screen-Sharing-Java/.gitignore new file mode 100644 index 00000000..93e62f94 --- /dev/null +++ b/Device-Screen-Sharing-Java/.gitignore @@ -0,0 +1,15 @@ +# intellij +*.iml + +.gradle +/local.properties +/.idea/workspace.xml +/.idea/libraries +.DS_Store +/build +/captures +.externalNativeBuild +app/build + +.settings/ +app/jniLibs/ diff --git a/Device-Screen-Sharing-Java/README.md b/Device-Screen-Sharing-Java/README.md new file mode 100644 index 00000000..a2cd565c --- /dev/null +++ b/Device-Screen-Sharing-Java/README.md @@ -0,0 +1,143 @@ +# Screen Sharing + +This app shows how to use `WebView` as the source for screen-sharing video. + +> Check [Basic-Video-Capturer-Camera-2](../Basic-Video-Capturer-Camera-2) project to see how a device camera can be used as the video source for the custom `Capturer`. +## Screen sharing + +Custom video capturer is using `WebView` from the Android application as the source of +a published stream. + +When the app starts up, the `onCreate` method instantiates a `WebView` object: + +```java +webViewContainer = findViewById(R.id.webview); +``` + +Upon connecting to the OpenTok session, the app instantiates a `Publisher` object, and calls its +`setCapturer` method to use a custom video capturer, defined by the `ScreenSharingCapturer` +class: + +```java +@Override +public void onConnected(Session session) { + ScreenSharingCapturer screenSharingCapturer = new ScreenSharingCapturer(MainActivity.this, webViewContainer); + + publisher = new Publisher.Builder(MainActivity.this) + .capturer(screenSharingCapturer) + .build(); + + publisher.setPublisherListener(publisherListener); + publisher.setPublisherVideoType(PublisherKit.PublisherKitVideoType.PublisherKitVideoTypeScreen); + publisher.setAudioFallbackEnabled(false); + + webViewContainer.setWebViewClient(new WebViewClient()); + WebSettings webSettings = webViewContainer.getSettings(); + webSettings.setJavaScriptEnabled(true); + webViewContainer.setLayerType(View.LAYER_TYPE_SOFTWARE, null); + webViewContainer.loadUrl("https://www.tokbox.com"); + + publisher.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL); + publisherViewContainer.addView(publisher.getView()); + + session.publish(publisher); +} +``` + +> Note: that the call to the `setPublisherVideoType` method sets the video type of the published +stream to `PublisherKitVideoType.PublisherKitVideoTypeScreen`. This optimizes the video encoding for +screen sharing. It is recommended to use a low frame rate (15 frames per second or lower) with this +video type. When using the screen video type in a session that uses the [OpenTok Media +Server](https://tokbox.com/opentok/tutorials/create-session/#media-mode), the +audio-only fallback feature is disabled, so that the video does not drop out in subscribers. + +The `onConnected` method also calls the `loadScreenWebView` method. This method +configures the WebView object, loading the TokBox URL. + +Note that the `webViewContainer` object is passed into the `ScreenSharingCapturer` constructor, +which assigns it to the `contentView` property. + +The `getCaptureSettings` method initializes capture settings to be used by the custom +video capturer: + +```java +@Override +public CaptureSettings getCaptureSettings() { + + CaptureSettings captureSettings = new CaptureSettings(); + captureSettings.fps = fps; + captureSettings.width = width; + captureSettings.height = height; + captureSettings.format = ARGB; + return captureSettings; +} +``` + +The `startCapture` method starts the `frameProducer` thread after 1/15 second: + +```java +@Override +public int startCapture() { + capturing = true; + + handler.postDelayed(newFrame, 1000 / fps); + return 0; +} +``` + +The `frameProducer` thread gets a `Bitmap` representation of the `contentView` object + (the `WebView`), writes its pixels to a buffer, and then calls the `provideIntArrayFrame()` + method, passing in that buffer as a parameter: + +```java +private Runnable newFrame = new Runnable() { + @Override + public void run() { + if (capturing) { + int width = contentView.getWidth(); + int height = contentView.getHeight(); + + if (frame == null || + ScreenSharingCapturer.this.width != width || + ScreenSharingCapturer.this.height != height) { + + ScreenSharingCapturer.this.width = width; + ScreenSharingCapturer.this.height = height; + + if (bmp != null) { + bmp.recycle(); + bmp = null; + } + + bmp = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); + + canvas = new Canvas(bmp); + frame = new int[width * height]; + } + canvas.saveLayer(0, 0, width, height, null); + canvas.translate(-contentView.getScrollX(), - contentView.getScrollY()); + contentView.draw(canvas); + + bmp.getPixels(frame, 0, width, 0, 0, width, height); + + provideIntArrayFrame(frame, ARGB, width, height, 0, false); + + canvas.restore(); + + handler.postDelayed(newFrame, 1000 / fps); + + } + } +}; +``` + +The `provideIntArrayFrame` method, defined by the `BaseVideoCapturer` class sends an integer array of data to the publisher, to be used for the next video frame published. + +If the publisher is still capturing video, the thread starts again after another 1/15 of a +second, so that the capturer continues to supply the publisher with new video frames to publish. +``` + +## Further Reading + +* Review [other sample projects](../) +* Read more about [OpenTok Android SDK](https://tokbox.com/developer/sdks/android/) diff --git a/Device-Screen-Sharing-Java/app/.gitignore b/Device-Screen-Sharing-Java/app/.gitignore new file mode 100644 index 00000000..796b96d1 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/.gitignore @@ -0,0 +1 @@ +/build diff --git a/Device-Screen-Sharing-Java/app/build.gradle b/Device-Screen-Sharing-Java/app/build.gradle new file mode 100644 index 00000000..bcec91f4 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/build.gradle @@ -0,0 +1,38 @@ +plugins { + id 'com.android.application' +} + +apply { + from '../../commons.gradle' +} + +android { + compileSdkVersion extCompileSdkVersion + + defaultConfig { + applicationId "com.tokbox.sample.screensharing" + minSdkVersion extMinSdkVersion + targetSdkVersion extTargetSdkVersion + versionCode extVersionCode + versionName extVersionName + } + + buildTypes { + release { + minifyEnabled extMinifyEnabled + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } + + compileOptions { + sourceCompatibility JavaVersion.VERSION_1_8 + targetCompatibility JavaVersion.VERSION_1_8 + } +} + +dependencies { + // Dependency versions are defined in the ../../commons.gradle file + implementation "com.opentok.android:opentok-android-sdk:${extOpentokSdkVersion}" + implementation "androidx.appcompat:appcompat:${extAppCompatVersion}" + implementation "pub.devrel:easypermissions:${extEasyPermissionsVersion}" +} diff --git a/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml b/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml new file mode 100644 index 00000000..6b6b8eb0 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml @@ -0,0 +1,29 @@ + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java new file mode 100644 index 00000000..66b1e7c2 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java @@ -0,0 +1,240 @@ +package com.tokbox.sample.screensharing; + +import android.Manifest; +import android.content.Context; +import android.content.Intent; +import android.media.projection.MediaProjection; +import android.media.projection.MediaProjectionManager; +import android.os.Build; +import android.os.Bundle; +import android.provider.MediaStore; +import android.util.Log; +import android.view.View; +import android.webkit.WebSettings; +import android.webkit.WebView; +import android.webkit.WebViewClient; +import android.widget.RelativeLayout; +import android.widget.Toast; +import androidx.annotation.NonNull; +import androidx.appcompat.app.AppCompatActivity; +import com.opentok.android.BaseVideoRenderer; +import com.opentok.android.OpentokError; +import com.opentok.android.Publisher; +import com.opentok.android.PublisherKit; +import com.opentok.android.Session; +import com.opentok.android.Stream; +import pub.devrel.easypermissions.AfterPermissionGranted; +import pub.devrel.easypermissions.EasyPermissions; +import java.util.List; + +public class MainActivity extends AppCompatActivity implements EasyPermissions.PermissionCallbacks { + + private static final String TAG = MainActivity.class.getSimpleName(); + private static final int REQUEST_MEDIA_PROJECTION = 100; + + private static final int PERMISSIONS_REQUEST_CODE = 124; + + private Session session; + private Publisher publisher; + + private RelativeLayout publisherViewContainer; + private WebView webViewContainer; + private ScreenSharingManager screenSharingManager; + private ScreenSharingCapturer screenSharingCapturer; + private MediaProjectionManager mediaProjectionManager; + private MediaProjection mediaProjection; + + private PublisherKit.PublisherListener publisherListener = new PublisherKit.PublisherListener() { + @Override + public void onStreamCreated(PublisherKit publisherKit, Stream stream) { + Log.d(TAG, "onStreamCreated: Own stream " + stream.getStreamId() + " created"); + } + + @Override + public void onStreamDestroyed(PublisherKit publisherKit, Stream stream) { + Log.d(TAG, "onStreamDestroyed: Own stream " + stream.getStreamId() + " destroyed"); + } + + @Override + public void onError(PublisherKit publisherKit, OpentokError opentokError) { + finishWithMessage("PublisherKit error: " + opentokError.getMessage()); + } + }; + + @Override + public void onActivityResult(int requestCode, int resultCode, Intent data) { + if (requestCode == REQUEST_MEDIA_PROJECTION) { + if (resultCode != AppCompatActivity.RESULT_OK || data == null) { + Toast.makeText( + this, + "screen_capture_permission_not_granted", + Toast.LENGTH_LONG) + .show(); + return; + } + screenSharingManager.startForeground(); + startScreenCapture(resultCode, data); + } + } + + private void requestScreenCapturePermission() { + Log.d(TAG, "Requesting permission to capture screen"); + mediaProjectionManager = + (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE); + + // This initiates a prompt dialog for the user to confirm screen projection. + startActivityForResult( + mediaProjectionManager.createScreenCaptureIntent(), REQUEST_MEDIA_PROJECTION); + } + + private void startScreenCapture(int resultCode, Intent data) { + mediaProjection = mediaProjectionManager.getMediaProjection(resultCode, data); + ScreenSharingCapturer screenSharingCapturer = new ScreenSharingCapturer(MainActivity.this, mediaProjection); + + publisher = new Publisher.Builder(MainActivity.this) + .capturer(screenSharingCapturer) + .build(); + + publisher.setPublisherListener(publisherListener); + publisher.setPublisherVideoType(PublisherKit.PublisherKitVideoType.PublisherKitVideoTypeScreen); + publisher.setAudioFallbackEnabled(false); + + publisher.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL); + publisherViewContainer.addView(publisher.getView()); + + session.publish(publisher); + } + + private Session.SessionListener sessionListener = new Session.SessionListener() { + @Override + public void onConnected(Session session) { + Log.d(TAG, "onConnected: Connected to session " + session.getSessionId()); + + if (Build.VERSION.SDK_INT >= 34) { + requestScreenCapturePermission(); + } else { + requestScreenCapturePermission(); + } + } + + @Override + public void onDisconnected(Session session) { + Log.d(TAG, "onDisconnected: disconnected from session " + session.getSessionId()); + + MainActivity.this.session = null; + } + + @Override + public void onError(Session session, OpentokError opentokError) { + finishWithMessage("Session error: " + opentokError.getMessage()); + } + + @Override + public void onStreamReceived(Session session, Stream stream) { + Log.d(TAG, "onStreamReceived: New stream " + stream.getStreamId() + " in session " + session.getSessionId()); + } + + @Override + public void onStreamDropped(Session session, Stream stream) { + Log.d(TAG, "onStreamDropped: Stream " + stream.getStreamId() + " dropped from session " + session.getSessionId()); + } + }; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_main); + screenSharingManager = new ScreenSharingManager(this); + + if(!OpenTokConfig.isValid()) { + finishWithMessage("Invalid OpenTokConfig. " + OpenTokConfig.getDescription()); + return; + } + + publisherViewContainer = findViewById(R.id.publisherview); + webViewContainer = findViewById(R.id.webview); + + requestPermissions(); + } + + @Override + protected void onPause() { + super.onPause(); + + if (session == null) { + return; + } + + session.onPause(); + + if (isFinishing()) { + disconnectSession(); + } + } + + @Override + protected void onResume() { + super.onResume(); + + if (session == null) { + return; + } + + session.onResume(); + } + + @Override + protected void onDestroy() { + disconnectSession(); + + super.onDestroy(); + } + + @Override + public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { + super.onRequestPermissionsResult(requestCode, permissions, grantResults); + EasyPermissions.onRequestPermissionsResult(requestCode, permissions, grantResults, this); + } + + @Override + public void onPermissionsGranted(int requestCode, List perms) { + Log.d(TAG, "onPermissionsGranted:" + requestCode + ": " + perms); + } + + @Override + public void onPermissionsDenied(int requestCode, List perms) { + finishWithMessage("onPermissionsDenied: " + requestCode + ": " + perms); + } + + @AfterPermissionGranted(PERMISSIONS_REQUEST_CODE) + private void requestPermissions() { + String[] perms = {Manifest.permission.INTERNET, Manifest.permission.CAMERA, Manifest.permission.RECORD_AUDIO, Manifest.permission.FOREGROUND_SERVICE}; + + if (EasyPermissions.hasPermissions(this, perms)) { + session = new Session.Builder(this, OpenTokConfig.API_KEY, OpenTokConfig.SESSION_ID).build(); + session.setSessionListener(sessionListener); + session.connect(OpenTokConfig.TOKEN); + } else { + EasyPermissions.requestPermissions(this, getString(R.string.rationale_video_app), PERMISSIONS_REQUEST_CODE, perms); + } + } + + private void disconnectSession() { + if (session == null) { + return; + } + + if (publisher != null) { + publisherViewContainer.removeView(publisher.getView()); + session.unpublish(publisher); + publisher = null; + } + session.disconnect(); + } + + private void finishWithMessage(String message) { + Log.e(TAG, message); + Toast.makeText(this, message, Toast.LENGTH_LONG).show(); + this.finish(); + } +} diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java new file mode 100644 index 00000000..143e370b --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java @@ -0,0 +1,38 @@ +package com.tokbox.sample.screensharing; + +import android.text.TextUtils; +import androidx.annotation.NonNull; + +public class OpenTokConfig { + /* + Fill the following variables using your own Project info from the OpenTok dashboard + https://dashboard.tokbox.com/projects + */ + + // Replace with a API key + public static final String API_KEY = "47521351"; + + // Replace with a generated Session ID + public static final String SESSION_ID = "1_MX40NzUyMTM1MX5-MTcxODM1OTEyMzUyNH5oaUhWbUx4TXJQdldMY2tmRGJ1bFB3a2Z-fn4"; + + // Replace with a generated token (from the dashboard or using an OpenTok server SDK) + public static final String TOKEN = "T1==cGFydG5lcl9pZD00NzUyMTM1MSZzaWc9OWU0YTYyZWVlMDk1YjY4YmFlNjhmZDRjZjkyYjU3ODU4MjcxZjQyNzpzZXNzaW9uX2lkPTFfTVg0ME56VXlNVE0xTVg1LU1UY3hPRE0xT1RFeU16VXlOSDVvYVVoV2JVeDRUWEpRZGxkTVkydG1SR0oxYkZCM2EyWi1mbjQmY3JlYXRlX3RpbWU9MTcxODM1OTEyOCZub25jZT0wLjAxMTc0MTEzNTQ5NDAxOTExNyZyb2xlPXB1Ymxpc2hlciZleHBpcmVfdGltZT0xNzIwOTUxMTI3JmluaXRpYWxfbGF5b3V0X2NsYXNzX2xpc3Q9"; + + public static boolean isValid() { + if (TextUtils.isEmpty(OpenTokConfig.API_KEY) + || TextUtils.isEmpty(OpenTokConfig.SESSION_ID) + || TextUtils.isEmpty(OpenTokConfig.TOKEN)) { + return false; + } + + return true; + } + + @NonNull + public static String getDescription() { + return "OpenTokConfig:" + "\n" + + "API_KEY: " + OpenTokConfig.API_KEY + "\n" + + "SESSION_ID: " + OpenTokConfig.SESSION_ID + "\n" + + "TOKEN: " + OpenTokConfig.TOKEN + "\n"; + } +} diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java new file mode 100644 index 00000000..c3be0e90 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java @@ -0,0 +1,170 @@ +package com.tokbox.sample.screensharing; + +import android.annotation.SuppressLint; +import android.content.Context; +import android.graphics.Bitmap; +import android.graphics.Canvas; +import android.graphics.PixelFormat; +import android.graphics.SurfaceTexture; +import android.hardware.display.DisplayManager; +import android.hardware.display.VirtualDisplay; +import android.media.Image; +import android.media.ImageReader; +import android.media.projection.MediaProjection; +import android.os.Handler; +import android.os.HandlerThread; +import android.util.DisplayMetrics; +import android.view.Surface; +import android.view.View; +import android.view.WindowManager; +import android.webkit.WebView; + +import java.nio.ByteBuffer; + +import com.opentok.android.BaseVideoCapturer; + +public class ScreenSharingCapturer extends BaseVideoCapturer { + + private MediaProjection mediaProjection; + private ImageReader imageReader; + private VirtualDisplay virtualDisplay; + private Handler backgroundHandler; + private HandlerThread backgroundThread; + + private Context context; + + private boolean capturing = false; + + private int fps = 15; + private int width = 0; + private int height = 0; + private int[] frame; + + public ScreenSharingCapturer(Context context, MediaProjection mediaProjection) { + this.context = context; + this.mediaProjection = mediaProjection; + initDisplayMetrics(); + } + + private void initDisplayMetrics() { + WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE); + if (windowManager != null) { + DisplayMetrics displayMetrics = new DisplayMetrics(); + windowManager.getDefaultDisplay().getMetrics(displayMetrics); + width = displayMetrics.widthPixels; + height = displayMetrics.heightPixels; + } + } + + @SuppressLint("WrongConstant") + @Override + public void init() { + imageReader = ImageReader.newInstance(width, height, PixelFormat.RGBA_8888, 2); + startBackgroundThread(); + } + + private void createVirtualDisplay() { + virtualDisplay = mediaProjection.createVirtualDisplay("ScreenCapture", + width, height, context.getResources().getDisplayMetrics().densityDpi, + DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, + imageReader.getSurface(), null, backgroundHandler); + + imageReader.setOnImageAvailableListener(new ImageReader.OnImageAvailableListener() { + @Override + public void onImageAvailable(ImageReader reader) { + Image image = reader.acquireLatestImage(); + if (image != null) { + Image.Plane[] planes = image.getPlanes(); + ByteBuffer buffer = planes[0].getBuffer(); + int pixelStride = planes[0].getPixelStride(); + int rowStride = planes[0].getRowStride(); + + if (frame == null) { + frame = new int[width * height]; + } + + for (int y = 0; y < height; y++) { + for (int x = 0; x < width; x++) { + int index = y * rowStride + x * pixelStride; + int pixel = buffer.getInt(index); + frame[y * width + x] = pixel; + } + } + + provideIntArrayFrame(frame, ARGB, width, height, 0, false); + image.close(); + } + } + }, backgroundHandler); + } + + @Override + public int startCapture() { + capturing = true; + return 0; + } + + @Override + public int stopCapture() { + capturing = false; + if (virtualDisplay != null) { + virtualDisplay.release(); + } + if (mediaProjection != null) { + mediaProjection.stop(); + } + stopBackgroundThread(); + return 0; + + } + + @Override + public boolean isCaptureStarted() { + return capturing; + } + + @Override + public CaptureSettings getCaptureSettings() { + + CaptureSettings captureSettings = new CaptureSettings(); + captureSettings.fps = fps; + captureSettings.width = width; + captureSettings.height = height; + captureSettings.format = ARGB; + return captureSettings; + } + + @Override + public void destroy() { + + } + + @Override + public void onPause() { + + } + + @Override + public void onResume() { + + } + + private void startBackgroundThread() { + createVirtualDisplay(); + backgroundThread = new HandlerThread("ScreenCapture"); + backgroundThread.start(); + backgroundHandler = new Handler(backgroundThread.getLooper()); + } + + private void stopBackgroundThread() { + backgroundThread.quitSafely(); + try { + backgroundThread.join(); + backgroundThread = null; + backgroundHandler = null; + } catch (InterruptedException e) { + e.printStackTrace(); + } + } + +} \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java new file mode 100644 index 00000000..efd22409 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java @@ -0,0 +1,65 @@ +package com.tokbox.sample.screensharing; + +import android.annotation.TargetApi; +import android.content.ComponentName; +import android.content.Context; +import android.content.Intent; +import android.content.ServiceConnection; +import android.os.IBinder; + +public class ScreenSharingManager { + private ScreenSharingService mService; + private Context mContext; + private State currentState = State.UNBIND_SERVICE; + + /** Defines callbacks for service binding, passed to bindService() */ + private ServiceConnection connection = + new ServiceConnection() { + + @Override + public void onServiceConnected(ComponentName className, IBinder service) { + // We've bound to ScreenCapturerService, cast the IBinder and get + // ScreenCapturerService instance + ScreenSharingService.LocalBinder binder = + (ScreenSharingService.LocalBinder) service; + mService = binder.getService(); + currentState = State.BIND_SERVICE; + } + + @Override + public void onServiceDisconnected(ComponentName arg0) {} + }; + + /** An enum describing the possible states of a ScreenCapturerManager. */ + public enum State { + BIND_SERVICE, + START_FOREGROUND, + END_FOREGROUND, + UNBIND_SERVICE + } + + ScreenSharingManager(Context context) { + mContext = context; + bindService(); + } + + private void bindService() { + Intent intent = new Intent(mContext, ScreenSharingService.class); + mContext.bindService(intent, connection, Context.BIND_AUTO_CREATE); + } + + void startForeground() { + mService.startForeground(); + currentState = State.START_FOREGROUND; + } + + void endForeground() { + mService.endForeground(); + currentState = State.END_FOREGROUND; + } + + void unbindService() { + mContext.unbindService(connection); + currentState = State.UNBIND_SERVICE; + } +} diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java new file mode 100644 index 00000000..935ef2e4 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java @@ -0,0 +1,73 @@ +package com.tokbox.sample.screensharing; + +import android.annotation.TargetApi; +import android.app.Notification; +import android.app.NotificationChannel; +import android.app.NotificationManager; +import android.app.Service; +import android.content.Context; +import android.content.Intent; +import android.os.Binder; +import android.os.IBinder; +import androidx.core.app.NotificationCompat; + +@TargetApi(29) +public class ScreenSharingService extends Service { + private static final String CHANNEL_ID = "screen_capture"; + private static final String CHANNEL_NAME = "Screen_Capture"; + + // Binder given to clients + private final IBinder binder = new LocalBinder(); + + /** + * Class used for the client Binder. We know this service always runs in the same process as its + * clients, we don't need to deal with IPC. + */ + public class LocalBinder extends Binder { + public ScreenSharingService getService() { + // Return this instance of ScreenCapturerService so clients can call public methods + return ScreenSharingService.this; + } + } + + @Override + public void onCreate() { + super.onCreate(); + } + + @Override + public int onStartCommand(Intent intent, int flags, int startId) { + return START_NOT_STICKY; + } + + public void startForeground() { + NotificationChannel chan = + new NotificationChannel( + CHANNEL_ID, CHANNEL_NAME, NotificationManager.IMPORTANCE_NONE); + NotificationManager manager = + (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); + assert manager != null; + manager.createNotificationChannel(chan); + + final int notificationId = (int) System.currentTimeMillis(); + NotificationCompat.Builder notificationBuilder = + new NotificationCompat.Builder(this, CHANNEL_ID); + Notification notification = + notificationBuilder + .setOngoing(true) + .setContentTitle("ScreenCapturerService is running in the foreground") + .setPriority(NotificationManager.IMPORTANCE_MIN) + .setCategory(Notification.CATEGORY_SERVICE) + .build(); + startForeground(notificationId, notification); + } + + public void endForeground() { + stopForeground(true); + } + + @Override + public IBinder onBind(Intent intent) { + return binder; + } +} \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml b/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml new file mode 100644 index 00000000..17ddb4c0 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml @@ -0,0 +1,34 @@ + + + + + + + + + + diff --git a/Device-Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png b/Device-Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..7dd6d427da11d23919c88c766b6c3b3c29354b19 GIT binary patch literal 2614 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 Ye~)Xyr$INTPXGV_07*qoM6N<$f+y%y=l}o! literal 0 HcmV?d00001 diff --git a/Device-Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png b/Device-Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..b1767f0a894943f8cb516696a4962f480fc5ebbe GIT binary patch literal 1442 zcmV;T1zq}yP)10L7{|{vlbB4>5*uS)YBh^2Txby$X``fq8$}Ab(WNf@ z7E%OJ@C&$8i+%tjSQjFws2DdY4G3Z*Hil|iV^Z6sd7DYS|I_?VpE+}HGB=sq&;yg_ zIsfPXyq%YOZ)xd9GTa?J4(wF>Ui&NB{{RUj`Kc-JQmGVli=)HC*UrT$+P}29xq0Uw zAy$u7mDI2}bELJ!x;PpZXO6VCSQkgb;>?lQO0`<;M!d};P7gMHa0dIc7X0| z!aVxz^e?9smbGf_ADm00?}+uZA8^B!cK;h6>~^Q_Kja?itN7hjv+>Bv4^MU$$?t## zAR!@Y-f?CMc+c{|-cI-CqXX{6k>1cqTH!W-TGDU_(wya>vz}#33)n37>y=ga(w8%C z{Oox*v$UR>BP%~OH8dcD{i&dxR-xIV(|D7Yue%eUPP=dCmb}^KrbzNz8(4S`U`$Iw zpfp=6u=&-hd-n4`LKAV!wESRx2Q;B0IcsizU~%A9&Nq@zvX{U5(_Pp)PHFkUs7Leg zIo<4Yy_H4Bv9uGfd@~!v4flguMGQyCdeNzs)?TFkLvkl3X8x-?0jfiTdN?hbhf55> zcIUme{t(1sMd@$+bg3b<&2ROF25fhV0vXv5xd7m|tLyIPe^-2D@=98rwBZP)8PhNv ziqRNccHlWqPkUnY?f9|W4eew96 z=gBZ)3%cn=UcCJ4u&yWBv>twF9QFDAaz@n5TfG|;yN@JEU) zUB7S#;*TZ>I7tpAtJ>_iyjpYbO#Ks@pP0GozMfld*4^i#206t#ff_2JV4SQR?kcp#cco1xu1m^JBdoZuC$G)WYQG z!FHd^E}rzXlM)^)alxuCZ5K{I+*erP1U`N29kl0c7gl6q<;uUXu&}2ch^sue zxaK~)xa=n9SKVClJv6sc^OV7^Hvf5`uTs4b^-gKUDA@+=iu?1m_5WjbMeHiUp^*LX zFO!raB*9{7ZSdhRw;XIpYlClXF@_D9<{zFy)_FSzc)Zvwmjo8GI=+&v@~`_RB?&kt z%(IFBN5a3I{yXjfMcE?JCDAP}#$>Y?x+MB;{lgB}RkJ~}aLbXN#mmB56+fV=S7)>i&#cHJ4z+!pRqy8@(i-N_{a?s&0 wmjnxU0MM>14lx$b2__la6eGss`Ck102U!@Vha~uIR{#J207*qoM6N<$f-@J?RR910 literal 0 HcmV?d00001 diff --git a/Device-Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/Device-Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..f284f639b85f96f7986e7c678fab9494f816aca8 GIT binary patch literal 3519 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 literal 0 HcmV?d00001 diff --git a/Device-Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/Device-Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..f277e80623b9b4d03b4d4192003208573279b0e4 GIT binary patch literal 6153 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 R)7xh;yJK~`=GMdK{||2;tFQn7 literal 0 HcmV?d00001 diff --git a/Device-Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/Device-Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..fa8a52414781599481e0240808659c8a93e7d65a GIT binary patch literal 9013 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 + + 64dp + diff --git a/Device-Screen-Sharing-Java/app/src/main/res/values/colors.xml b/Device-Screen-Sharing-Java/app/src/main/res/values/colors.xml new file mode 100644 index 00000000..3ab3e9cb --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/res/values/colors.xml @@ -0,0 +1,6 @@ + + + #3F51B5 + #303F9F + #FF4081 + diff --git a/Device-Screen-Sharing-Java/app/src/main/res/values/dimens.xml b/Device-Screen-Sharing-Java/app/src/main/res/values/dimens.xml new file mode 100644 index 00000000..47c82246 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/res/values/dimens.xml @@ -0,0 +1,5 @@ + + + 16dp + 16dp + diff --git a/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml b/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml new file mode 100644 index 00000000..7367ce56 --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml @@ -0,0 +1,4 @@ + + Screen-Sharing + This app needs access to your camera and mic so you can perform video calls + diff --git a/Device-Screen-Sharing-Java/app/src/main/res/values/styles.xml b/Device-Screen-Sharing-Java/app/src/main/res/values/styles.xml new file mode 100644 index 00000000..5885930d --- /dev/null +++ b/Device-Screen-Sharing-Java/app/src/main/res/values/styles.xml @@ -0,0 +1,11 @@ + + + + + + diff --git a/Device-Screen-Sharing-Java/build.gradle b/Device-Screen-Sharing-Java/build.gradle new file mode 100644 index 00000000..2fcc7ef7 --- /dev/null +++ b/Device-Screen-Sharing-Java/build.gradle @@ -0,0 +1,9 @@ +// Top-level build file where you can add configuration options common to all sub-projects/modules. +plugins { + id 'com.android.application' version '7.1.2' apply false + id 'com.android.library' version '7.1.2' apply false +} + +task clean(type: Delete) { + delete rootProject.buildDir +} \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/gradle.properties b/Device-Screen-Sharing-Java/gradle.properties new file mode 100644 index 00000000..a48987c4 --- /dev/null +++ b/Device-Screen-Sharing-Java/gradle.properties @@ -0,0 +1,20 @@ +# Project-wide Gradle settings. + +# IDE (e.g. Android Studio) users: +# Gradle settings configured through the IDE *will override* +# any settings specified in this file. + +# For more details on how to configure your build environment visit +# http://www.gradle.org/docs/current/userguide/build_environment.html + +# Specifies the JVM arguments used for the daemon process. +# The setting is particularly useful for tweaking memory settings. +org.gradle.jvmargs=-Xmx4096m + +# When configured, Gradle will run in incubating parallel mode. +# This option should only be used with decoupled projects. More details, visit +# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects +# org.gradle.parallel=true +android.useAndroidX=true +android.enableJetifier=true +org.gradle.parallel=true \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar b/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..13372aef5e24af05341d49695ee84e5f9b594659 GIT binary patch literal 53636 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ⅅ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 H-+%o-(+5aQ literal 0 HcmV?d00001 diff --git a/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties b/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 00000000..9c6be48d --- /dev/null +++ b/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +#Mon May 04 11:02:47 CEST 2020 +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-7.2-bin.zip diff --git a/Device-Screen-Sharing-Java/gradlew b/Device-Screen-Sharing-Java/gradlew new file mode 100755 index 00000000..d8c4edf1 --- /dev/null +++ b/Device-Screen-Sharing-Java/gradlew @@ -0,0 +1,160 @@ +#!/usr/bin/env bash + +############################################################################## +## +## Gradle start up script for UN*X +## +############################################################################## + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS="" + +APP_NAME="Gradle" +APP_BASE_NAME=`basename "$0"` + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD="maximum" + +warn ( ) { + echo "$*" +} + +die ( ) { + echo + echo "$*" + echo + exit 1 +} + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +case "`uname`" in + CYGWIN* ) + cygwin=true + ;; + Darwin* ) + darwin=true + ;; + MINGW* ) + msys=true + ;; +esac + +# Attempt to set APP_HOME +# Resolve links: $0 may be a link +PRG="$0" +# Need this for relative symlinks. +while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" >/dev/null +APP_HOME="`pwd -P`" +cd "$SAVED" >/dev/null + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation" + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation" +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin, switch paths to Windows format before running java +if $cygwin ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + JAVACMD=`cygpath --unix "$JAVACMD"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=$((i+1)) + done + case $i in + (0) set -- ;; + (1) set -- "$args0" ;; + (2) set -- "$args0" "$args1" ;; + (3) set -- "$args0" "$args1" "$args2" ;; + (4) set -- "$args0" "$args1" "$args2" "$args3" ;; + (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules +function splitJvmOpts() { + JVM_OPTS=("$@") +} +eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS +JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" + +exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" diff --git a/Device-Screen-Sharing-Java/gradlew.bat b/Device-Screen-Sharing-Java/gradlew.bat new file mode 100644 index 00000000..aec99730 --- /dev/null +++ b/Device-Screen-Sharing-Java/gradlew.bat @@ -0,0 +1,90 @@ +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS= + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto init + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto init + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:init +@rem Get command-line arguments, handling Windowz variants + +if not "%OS%" == "Windows_NT" goto win9xME_args +if "%@eval[2+2]" == "4" goto 4NT_args + +:win9xME_args +@rem Slurp the command line arguments. +set CMD_LINE_ARGS= +set _SKIP=2 + +:win9xME_args_slurp +if "x%~1" == "x" goto execute + +set CMD_LINE_ARGS=%* +goto execute + +:4NT_args +@rem Get arguments from the 4NT Shell from JP Software +set CMD_LINE_ARGS=%$ + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/Device-Screen-Sharing-Java/settings.gradle b/Device-Screen-Sharing-Java/settings.gradle new file mode 100644 index 00000000..f60920c6 --- /dev/null +++ b/Device-Screen-Sharing-Java/settings.gradle @@ -0,0 +1,16 @@ +pluginManagement { + repositories { + gradlePluginPortal() + google() + mavenCentral() + } +} +dependencyResolutionManagement { + repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) + repositories { + google() + mavenCentral() + } +} + +include ':app' From 2f6d7c954b77eaeb644fb0dbbbf6b61c3846e613 Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Wed, 1 Oct 2025 12:06:13 +0200 Subject: [PATCH 2/9] Screen sharing working for api level 35 --- Device-Screen-Sharing-Java/app/build.gradle | 2 + .../app/src/main/AndroidManifest.xml | 6 +- .../sample/screensharing/MainActivity.java | 120 +++++-- .../sample/screensharing/OpenTokConfig.java | 6 +- .../screensharing/ScreenSharingCapturer.java | 29 +- .../screensharing/ScreenSharingService.java | 50 +-- .../app/src/main/res/layout/activity_main.xml | 59 ++-- Device-Screen-Sharing-Java/build.gradle | 4 +- .../gradle/wrapper/gradle-wrapper.jar | Bin 53636 -> 43462 bytes .../gradle/wrapper/gradle-wrapper.properties | 5 +- Device-Screen-Sharing-Java/gradlew | 311 +++++++++++------- Device-Screen-Sharing-Java/gradlew.bat | 66 ++-- 12 files changed, 421 insertions(+), 237 deletions(-) diff --git a/Device-Screen-Sharing-Java/app/build.gradle b/Device-Screen-Sharing-Java/app/build.gradle index bcec91f4..904966c8 100644 --- a/Device-Screen-Sharing-Java/app/build.gradle +++ b/Device-Screen-Sharing-Java/app/build.gradle @@ -7,6 +7,7 @@ apply { } android { + namespace "com.tokbox.sample.screensharing" compileSdkVersion extCompileSdkVersion defaultConfig { @@ -35,4 +36,5 @@ dependencies { implementation "com.opentok.android:opentok-android-sdk:${extOpentokSdkVersion}" implementation "androidx.appcompat:appcompat:${extAppCompatVersion}" implementation "pub.devrel:easypermissions:${extEasyPermissionsVersion}" + implementation "androidx.constraintlayout:constraintlayout:${extConstraintLyoutVersion}" } diff --git a/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml b/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml index 6b6b8eb0..b78e7e5f 100644 --- a/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml +++ b/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml @@ -6,6 +6,7 @@ + @@ -20,10 +21,9 @@ - + android:foregroundServiceType="mediaProjection" + android:exported="false" /> \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java index 66b1e7c2..f10c16ec 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java @@ -3,16 +3,19 @@ import android.Manifest; import android.content.Context; import android.content.Intent; +import android.content.pm.ServiceInfo; import android.media.projection.MediaProjection; import android.media.projection.MediaProjectionManager; import android.os.Build; import android.os.Bundle; +import android.os.Handler; import android.provider.MediaStore; import android.util.Log; import android.view.View; import android.webkit.WebSettings; import android.webkit.WebView; import android.webkit.WebViewClient; +import android.widget.FrameLayout; import android.widget.RelativeLayout; import android.widget.Toast; import androidx.annotation.NonNull; @@ -23,22 +26,26 @@ import com.opentok.android.PublisherKit; import com.opentok.android.Session; import com.opentok.android.Stream; +import com.opentok.android.Subscriber; +import com.opentok.android.SubscriberKit; + import pub.devrel.easypermissions.AfterPermissionGranted; import pub.devrel.easypermissions.EasyPermissions; import java.util.List; +import java.util.Objects; public class MainActivity extends AppCompatActivity implements EasyPermissions.PermissionCallbacks { private static final String TAG = MainActivity.class.getSimpleName(); private static final int REQUEST_MEDIA_PROJECTION = 100; - private static final int PERMISSIONS_REQUEST_CODE = 124; - private Session session; private Publisher publisher; + private Subscriber subscriber; + + private FrameLayout publisherViewContainer; + private FrameLayout subscriberViewContainer; - private RelativeLayout publisherViewContainer; - private WebView webViewContainer; private ScreenSharingManager screenSharingManager; private ScreenSharingCapturer screenSharingCapturer; private MediaProjectionManager mediaProjectionManager; @@ -72,38 +79,44 @@ public void onActivityResult(int requestCode, int resultCode, Intent data) { .show(); return; } - screenSharingManager.startForeground(); startScreenCapture(resultCode, data); } } private void requestScreenCapturePermission() { Log.d(TAG, "Requesting permission to capture screen"); - mediaProjectionManager = - (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE); + mediaProjectionManager = (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE); - // This initiates a prompt dialog for the user to confirm screen projection. startActivityForResult( - mediaProjectionManager.createScreenCaptureIntent(), REQUEST_MEDIA_PROJECTION); + mediaProjectionManager.createScreenCaptureIntent(), REQUEST_MEDIA_PROJECTION); } private void startScreenCapture(int resultCode, Intent data) { - mediaProjection = mediaProjectionManager.getMediaProjection(resultCode, data); - ScreenSharingCapturer screenSharingCapturer = new ScreenSharingCapturer(MainActivity.this, mediaProjection); - - publisher = new Publisher.Builder(MainActivity.this) - .capturer(screenSharingCapturer) - .build(); - - publisher.setPublisherListener(publisherListener); - publisher.setPublisherVideoType(PublisherKit.PublisherKitVideoType.PublisherKitVideoTypeScreen); - publisher.setAudioFallbackEnabled(false); - - publisher.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL); - publisherViewContainer.addView(publisher.getView()); - - session.publish(publisher); + if (screenSharingManager != null) { + // Ensure the service is bound + screenSharingManager.startForeground(); // this calls startForeground() in the service + + // Wait until the service is started in the foreground before requesting MediaProjection + new Handler().postDelayed(() -> { + mediaProjection = mediaProjectionManager.getMediaProjection(resultCode, data); + ScreenSharingCapturer screenSharingCapturer = + new ScreenSharingCapturer(MainActivity.this, mediaProjection); + + publisher = new Publisher.Builder(MainActivity.this) + .capturer(screenSharingCapturer) + .build(); + + publisher.setPublisherListener(publisherListener); + publisher.setPublisherVideoType(PublisherKit.PublisherKitVideoType.PublisherKitVideoTypeScreen); + publisher.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL); + + publisherViewContainer.addView(publisher.getView()); + session.publish(publisher); + + }, 100); // small delay to ensure startForeground() has taken effect + } } + private Session.SessionListener sessionListener = new Session.SessionListener() { @Override @@ -132,11 +145,41 @@ public void onError(Session session, OpentokError opentokError) { @Override public void onStreamReceived(Session session, Stream stream) { Log.d(TAG, "onStreamReceived: New stream " + stream.getStreamId() + " in session " + session.getSessionId()); + + if (subscriber == null) { + subscriber = new Subscriber.Builder(MainActivity.this, stream).build(); + subscriber.getRenderer().setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL); + subscriber.setSubscriberListener(subscriberListener); + session.subscribe(subscriber); + subscriberViewContainer.addView(subscriber.getView()); + } } @Override public void onStreamDropped(Session session, Stream stream) { Log.d(TAG, "onStreamDropped: Stream " + stream.getStreamId() + " dropped from session " + session.getSessionId()); + + if (subscriber != null) { + subscriber = null; + subscriberViewContainer.removeAllViews(); + } + } + }; + + SubscriberKit.SubscriberListener subscriberListener = new SubscriberKit.SubscriberListener() { + @Override + public void onConnected(SubscriberKit subscriberKit) { + Log.d(TAG, "onConnected: Subscriber connected. Stream: " + subscriberKit.getStream().getStreamId()); + } + + @Override + public void onDisconnected(SubscriberKit subscriberKit) { + Log.d(TAG, "onDisconnected: Subscriber disconnected. Stream: " + subscriberKit.getStream().getStreamId()); + } + + @Override + public void onError(SubscriberKit subscriberKit, OpentokError opentokError) { + finishWithMessage("SubscriberKit onError: " + opentokError.getMessage()); } }; @@ -151,8 +194,8 @@ protected void onCreate(Bundle savedInstanceState) { return; } - publisherViewContainer = findViewById(R.id.publisherview); - webViewContainer = findViewById(R.id.webview); + publisherViewContainer = findViewById(R.id.publisher_container); + subscriberViewContainer = findViewById(R.id.subscriber_container); requestPermissions(); } @@ -199,6 +242,10 @@ public void onRequestPermissionsResult(int requestCode, @NonNull String[] permis @Override public void onPermissionsGranted(int requestCode, List perms) { Log.d(TAG, "onPermissionsGranted:" + requestCode + ": " + perms); + + if (requestCode == PERMISSIONS_REQUEST_CODE) { + initializeSession(OpenTokConfig.API_KEY, OpenTokConfig.SESSION_ID, OpenTokConfig.TOKEN); + } } @Override @@ -208,17 +255,30 @@ public void onPermissionsDenied(int requestCode, List perms) { @AfterPermissionGranted(PERMISSIONS_REQUEST_CODE) private void requestPermissions() { - String[] perms = {Manifest.permission.INTERNET, Manifest.permission.CAMERA, Manifest.permission.RECORD_AUDIO, Manifest.permission.FOREGROUND_SERVICE}; + String[] perms = {Manifest.permission.INTERNET, Manifest.permission.CAMERA, Manifest.permission.RECORD_AUDIO, Manifest.permission.FOREGROUND_SERVICE, Manifest.permission.FOREGROUND_SERVICE_MEDIA_PROJECTION}; if (EasyPermissions.hasPermissions(this, perms)) { - session = new Session.Builder(this, OpenTokConfig.API_KEY, OpenTokConfig.SESSION_ID).build(); - session.setSessionListener(sessionListener); - session.connect(OpenTokConfig.TOKEN); + initializeSession(OpenTokConfig.API_KEY, OpenTokConfig.SESSION_ID, OpenTokConfig.TOKEN); } else { EasyPermissions.requestPermissions(this, getString(R.string.rationale_video_app), PERMISSIONS_REQUEST_CODE, perms); } } + private void initializeSession(String apiKey, String sessionId, String token) { + Log.i(TAG, "apiKey: " + apiKey); + Log.i(TAG, "sessionId: " + sessionId); + Log.i(TAG, "token: " + token); + + /* + The context used depends on the specific use case, but usually, it is desired for the session to + live outside of the Activity e.g: live between activities. For a production applications, + it's convenient to use Application context instead of Activity context. + */ + session = new Session.Builder(this, apiKey, sessionId).build(); + session.setSessionListener(sessionListener); + session.connect(token); + } + private void disconnectSession() { if (session == null) { return; diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java index 143e370b..0af57414 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java @@ -10,13 +10,13 @@ public class OpenTokConfig { */ // Replace with a API key - public static final String API_KEY = "47521351"; + public static final String API_KEY = ""; // Replace with a generated Session ID - public static final String SESSION_ID = "1_MX40NzUyMTM1MX5-MTcxODM1OTEyMzUyNH5oaUhWbUx4TXJQdldMY2tmRGJ1bFB3a2Z-fn4"; + public static final String SESSION_ID = ""; // Replace with a generated token (from the dashboard or using an OpenTok server SDK) - public static final String TOKEN = "T1==cGFydG5lcl9pZD00NzUyMTM1MSZzaWc9OWU0YTYyZWVlMDk1YjY4YmFlNjhmZDRjZjkyYjU3ODU4MjcxZjQyNzpzZXNzaW9uX2lkPTFfTVg0ME56VXlNVE0xTVg1LU1UY3hPRE0xT1RFeU16VXlOSDVvYVVoV2JVeDRUWEpRZGxkTVkydG1SR0oxYkZCM2EyWi1mbjQmY3JlYXRlX3RpbWU9MTcxODM1OTEyOCZub25jZT0wLjAxMTc0MTEzNTQ5NDAxOTExNyZyb2xlPXB1Ymxpc2hlciZleHBpcmVfdGltZT0xNzIwOTUxMTI3JmluaXRpYWxfbGF5b3V0X2NsYXNzX2xpc3Q9"; + public static final String TOKEN = ""; public static boolean isValid() { if (TextUtils.isEmpty(OpenTokConfig.API_KEY) diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java index c3be0e90..88a8ce76 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java @@ -13,6 +13,7 @@ import android.media.projection.MediaProjection; import android.os.Handler; import android.os.HandlerThread; +import android.os.Looper; import android.util.DisplayMetrics; import android.view.Surface; import android.view.View; @@ -64,11 +65,27 @@ public void init() { } private void createVirtualDisplay() { - virtualDisplay = mediaProjection.createVirtualDisplay("ScreenCapture", - width, height, context.getResources().getDisplayMetrics().densityDpi, - DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, - imageReader.getSurface(), null, backgroundHandler); + mediaProjection.registerCallback(new MediaProjection.Callback() { + @Override + public void onStop() { + // MediaProjection was stopped, release resources + if (virtualDisplay != null) { + virtualDisplay.release(); + virtualDisplay = null; + } + } + }, new Handler(Looper.getMainLooper())); + + virtualDisplay = mediaProjection.createVirtualDisplay( + "ScreenSharing", + width, height, context.getResources().getDisplayMetrics().densityDpi, + DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, + imageReader.getSurface(), + null, + backgroundHandler + ); + imageReader.setOnImageAvailableListener(new ImageReader.OnImageAvailableListener() { @Override public void onImageAvailable(ImageReader reader) { @@ -91,7 +108,7 @@ public void onImageAvailable(ImageReader reader) { } } - provideIntArrayFrame(frame, ARGB, width, height, 0, false); + provideIntArrayFrame(frame, ABGR, width, height, 0, false); image.close(); } } @@ -130,7 +147,7 @@ public CaptureSettings getCaptureSettings() { captureSettings.fps = fps; captureSettings.width = width; captureSettings.height = height; - captureSettings.format = ARGB; + captureSettings.format = ABGR; return captureSettings; } diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java index 935ef2e4..26ab69ff 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java @@ -7,7 +7,9 @@ import android.app.Service; import android.content.Context; import android.content.Intent; +import android.content.pm.ServiceInfo; import android.os.Binder; +import android.os.Build; import android.os.IBinder; import androidx.core.app.NotificationCompat; @@ -15,6 +17,8 @@ public class ScreenSharingService extends Service { private static final String CHANNEL_ID = "screen_capture"; private static final String CHANNEL_NAME = "Screen_Capture"; + private static final int NOTIFICATION_ID = 123; + Notification notification; // Binder given to clients private final IBinder binder = new LocalBinder(); @@ -33,33 +37,41 @@ public ScreenSharingService getService() { @Override public void onCreate() { super.onCreate(); + createNotificationChannel(); + notification = createNotification(); + } + + private void createNotificationChannel() { + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { + NotificationChannel channel = new NotificationChannel( + CHANNEL_ID, + "Screen Sharing", + NotificationManager.IMPORTANCE_DEFAULT + ); + NotificationManager manager = getSystemService(NotificationManager.class); + if (manager != null) { + manager.createNotificationChannel(channel); + } + } + } + + private Notification createNotification() { + return new NotificationCompat.Builder(this, CHANNEL_ID) + .setContentTitle("Screen Sharing") + .setContentText("Screen sharing is active") + .setSmallIcon(R.mipmap.ic_launcher) + .setPriority(NotificationCompat.PRIORITY_DEFAULT) + .build(); } @Override public int onStartCommand(Intent intent, int flags, int startId) { + startForeground(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PROJECTION); return START_NOT_STICKY; } public void startForeground() { - NotificationChannel chan = - new NotificationChannel( - CHANNEL_ID, CHANNEL_NAME, NotificationManager.IMPORTANCE_NONE); - NotificationManager manager = - (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); - assert manager != null; - manager.createNotificationChannel(chan); - - final int notificationId = (int) System.currentTimeMillis(); - NotificationCompat.Builder notificationBuilder = - new NotificationCompat.Builder(this, CHANNEL_ID); - Notification notification = - notificationBuilder - .setOngoing(true) - .setContentTitle("ScreenCapturerService is running in the foreground") - .setPriority(NotificationManager.IMPORTANCE_MIN) - .setCategory(Notification.CATEGORY_SERVICE) - .build(); - startForeground(notificationId, notification); + startForeground(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PROJECTION); } public void endForeground() { diff --git a/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml b/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml index 17ddb4c0..7d2f43ac 100644 --- a/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml +++ b/Device-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml @@ -1,34 +1,35 @@ - + - - - + app:layout_constraintBottom_toBottomOf="parent" + app:layout_constraintEnd_toEndOf="parent" + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toTopOf="parent"> + + + + + + - - - + \ No newline at end of file diff --git a/Device-Screen-Sharing-Java/build.gradle b/Device-Screen-Sharing-Java/build.gradle index 2fcc7ef7..540c87f1 100644 --- a/Device-Screen-Sharing-Java/build.gradle +++ b/Device-Screen-Sharing-Java/build.gradle @@ -1,7 +1,7 @@ // Top-level build file where you can add configuration options common to all sub-projects/modules. plugins { - id 'com.android.application' version '7.1.2' apply false - id 'com.android.library' version '7.1.2' apply false + id 'com.android.application' version '8.6.0' apply false + id 'com.android.library' version '8.6.0' apply false } task clean(type: Delete) { diff --git a/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar b/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar index 13372aef5e24af05341d49695ee84e5f9b594659..d64cd4917707c1f8861d8cb53dd15194d4248596 100644 GIT binary patch literal 43462 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 literal 53636 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ⅅ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 H-+%o-(+5aQ diff --git a/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties b/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties index 9c6be48d..b82aa23a 100644 --- a/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties +++ b/Device-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,7 @@ -#Mon May 04 11:02:47 CEST 2020 distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.7-bin.zip +networkTimeout=10000 +validateDistributionUrl=true zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-7.2-bin.zip diff --git a/Device-Screen-Sharing-Java/gradlew b/Device-Screen-Sharing-Java/gradlew index d8c4edf1..1aa94a42 100755 --- a/Device-Screen-Sharing-Java/gradlew +++ b/Device-Screen-Sharing-Java/gradlew @@ -1,160 +1,249 @@ -#!/usr/bin/env bash +#!/bin/sh + +# +# Copyright © 2015-2021 the original authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# ############################################################################## -## -## Gradle start up script for UN*X -## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# ############################################################################## -# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -DEFAULT_JVM_OPTS="" +# Attempt to set APP_HOME + +# Resolve links: $0 may be a link +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac +done -APP_NAME="Gradle" -APP_BASE_NAME=`basename "$0"` +# This is normally unused +# shellcheck disable=SC2034 +APP_BASE_NAME=${0##*/} +# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) +APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit # Use the maximum available, or set MAX_FD != -1 to use that value. -MAX_FD="maximum" +MAX_FD=maximum -warn ( ) { +warn () { echo "$*" -} +} >&2 -die ( ) { +die () { echo echo "$*" echo exit 1 -} +} >&2 # OS specific support (must be 'true' or 'false'). cygwin=false msys=false darwin=false -case "`uname`" in - CYGWIN* ) - cygwin=true - ;; - Darwin* ) - darwin=true - ;; - MINGW* ) - msys=true - ;; +nonstop=false +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; esac -# Attempt to set APP_HOME -# Resolve links: $0 may be a link -PRG="$0" -# Need this for relative symlinks. -while [ -h "$PRG" ] ; do - ls=`ls -ld "$PRG"` - link=`expr "$ls" : '.*-> \(.*\)$'` - if expr "$link" : '/.*' > /dev/null; then - PRG="$link" - else - PRG=`dirname "$PRG"`"/$link" - fi -done -SAVED="`pwd`" -cd "`dirname \"$PRG\"`/" >/dev/null -APP_HOME="`pwd -P`" -cd "$SAVED" >/dev/null - CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + # Determine the Java command to use to start the JVM. if [ -n "$JAVA_HOME" ] ; then if [ -x "$JAVA_HOME/jre/sh/java" ] ; then # IBM's JDK on AIX uses strange locations for the executables - JAVACMD="$JAVA_HOME/jre/sh/java" + JAVACMD=$JAVA_HOME/jre/sh/java else - JAVACMD="$JAVA_HOME/bin/java" + JAVACMD=$JAVA_HOME/bin/java fi if [ ! -x "$JAVACMD" ] ; then die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME Please set the JAVA_HOME variable in your environment to match the -location of your Java installation" +location of your Java installation." fi else - JAVACMD="java" - which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + JAVACMD=java + if ! command -v java >/dev/null 2>&1 + then + die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. Please set the JAVA_HOME variable in your environment to match the -location of your Java installation" +location of your Java installation." + fi fi # Increase the maximum file descriptors if we can. -if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then - MAX_FD_LIMIT=`ulimit -H -n` - if [ $? -eq 0 ] ; then - if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then - MAX_FD="$MAX_FD_LIMIT" - fi - ulimit -n $MAX_FD - if [ $? -ne 0 ] ; then - warn "Could not set maximum file descriptor limit: $MAX_FD" - fi - else - warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" - fi +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC2039,SC3045 + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac fi -# For Darwin, add options to specify how the application appears in the dock -if $darwin; then - GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" -fi +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. + +# For Cygwin or MSYS, switch paths to Windows format before running java +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) -# For Cygwin, switch paths to Windows format before running java -if $cygwin ; then - APP_HOME=`cygpath --path --mixed "$APP_HOME"` - CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` - JAVACMD=`cygpath --unix "$JAVACMD"` - - # We build the pattern for arguments to be converted via cygpath - ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` - SEP="" - for dir in $ROOTDIRSRAW ; do - ROOTDIRS="$ROOTDIRS$SEP$dir" - SEP="|" - done - OURCYGPATTERN="(^($ROOTDIRS))" - # Add a user-defined pattern to the cygpath arguments - if [ "$GRADLE_CYGPATTERN" != "" ] ; then - OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" - fi # Now convert the arguments - kludge to limit ourselves to /bin/sh - i=0 - for arg in "$@" ; do - CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` - CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option - - if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition - eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` - else - eval `echo args$i`="\"$arg\"" + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) fi - i=$((i+1)) + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg done - case $i in - (0) set -- ;; - (1) set -- "$args0" ;; - (2) set -- "$args0" "$args1" ;; - (3) set -- "$args0" "$args1" "$args2" ;; - (4) set -- "$args0" "$args1" "$args2" "$args3" ;; - (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; - (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; - (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; - (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; - (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; - esac fi -# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules -function splitJvmOpts() { - JVM_OPTS=("$@") -} -eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS -JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" -exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Collect all arguments for the java command: +# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, +# and any embedded shellness will be escaped. +# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be +# treated as '${Hostname}' itself on the command line. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# + +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' + +exec "$JAVACMD" "$@" diff --git a/Device-Screen-Sharing-Java/gradlew.bat b/Device-Screen-Sharing-Java/gradlew.bat index aec99730..6689b85b 100644 --- a/Device-Screen-Sharing-Java/gradlew.bat +++ b/Device-Screen-Sharing-Java/gradlew.bat @@ -1,4 +1,20 @@ -@if "%DEBUG%" == "" @echo off +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + +@if "%DEBUG%"=="" @echo off @rem ########################################################################## @rem @rem Gradle startup script for Windows @@ -8,20 +24,24 @@ @rem Set local scope for the variables with windows NT shell if "%OS%"=="Windows_NT" setlocal -@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -set DEFAULT_JVM_OPTS= - set DIRNAME=%~dp0 -if "%DIRNAME%" == "" set DIRNAME=. +if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + @rem Find java.exe if defined JAVA_HOME goto findJavaFromJavaHome set JAVA_EXE=java.exe %JAVA_EXE% -version >NUL 2>&1 -if "%ERRORLEVEL%" == "0" goto init +if %ERRORLEVEL% equ 0 goto execute echo. echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. @@ -35,7 +55,7 @@ goto fail set JAVA_HOME=%JAVA_HOME:"=% set JAVA_EXE=%JAVA_HOME%/bin/java.exe -if exist "%JAVA_EXE%" goto init +if exist "%JAVA_EXE%" goto execute echo. echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% @@ -45,44 +65,26 @@ echo location of your Java installation. goto fail -:init -@rem Get command-line arguments, handling Windowz variants - -if not "%OS%" == "Windows_NT" goto win9xME_args -if "%@eval[2+2]" == "4" goto 4NT_args - -:win9xME_args -@rem Slurp the command line arguments. -set CMD_LINE_ARGS= -set _SKIP=2 - -:win9xME_args_slurp -if "x%~1" == "x" goto execute - -set CMD_LINE_ARGS=%* -goto execute - -:4NT_args -@rem Get arguments from the 4NT Shell from JP Software -set CMD_LINE_ARGS=%$ - :execute @rem Setup the command line set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + @rem Execute Gradle -"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* :end @rem End local scope for the variables with windows NT shell -if "%ERRORLEVEL%"=="0" goto mainEnd +if %ERRORLEVEL% equ 0 goto mainEnd :fail rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of rem the _cmd.exe /c_ return code! -if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 -exit /b 1 +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% :mainEnd if "%OS%"=="Windows_NT" endlocal From c189ec4a86ccb78456f139e7b18f43f150313db2 Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Wed, 1 Oct 2025 12:14:52 +0200 Subject: [PATCH 3/9] Fix consitencies --- ...build-basic-device-screen-sharing-java.yml | 25 +++++++++++++++++++ Device-Screen-Sharing-Java/app/build.gradle | 4 +-- .../app/src/main/AndroidManifest.xml | 2 +- .../MainActivity.java | 2 +- .../OpenTokConfig.java | 2 +- .../ScreenSharingCapturer.java | 2 +- .../ScreenSharingManager.java | 2 +- .../ScreenSharingService.java | 2 +- 8 files changed, 33 insertions(+), 8 deletions(-) create mode 100644 .github/workflows/build-basic-device-screen-sharing-java.yml rename Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/{screensharing => devicescreensharing}/MainActivity.java (99%) rename Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/{screensharing => devicescreensharing}/OpenTokConfig.java (95%) rename Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/{screensharing => devicescreensharing}/ScreenSharingCapturer.java (99%) rename Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/{screensharing => devicescreensharing}/ScreenSharingManager.java (97%) rename Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/{screensharing => devicescreensharing}/ScreenSharingService.java (98%) diff --git a/.github/workflows/build-basic-device-screen-sharing-java.yml b/.github/workflows/build-basic-device-screen-sharing-java.yml new file mode 100644 index 00000000..db265d6d --- /dev/null +++ b/.github/workflows/build-basic-device-screen-sharing-java.yml @@ -0,0 +1,25 @@ +name: Build Basic-Device-Screen-Sharing-Java + +on: + push: + branches: [main] # Just in case main was not up to date while merging PR + pull_request: + types: [opened, synchronize] + +jobs: + run: + continue-on-error: true + runs-on: ubuntu-latest + strategy: + fail-fast: false + steps: + - name: checkout + uses: actions/checkout@v2 + + - name: Set up JDK + uses: actions/setup-java@v1 + with: + java-version: 17 + + - name: Build + run: cd Basic-Device-Screen-Sharing-Java && ./gradlew app:assembleRelease && cd .. diff --git a/Device-Screen-Sharing-Java/app/build.gradle b/Device-Screen-Sharing-Java/app/build.gradle index 904966c8..0215b086 100644 --- a/Device-Screen-Sharing-Java/app/build.gradle +++ b/Device-Screen-Sharing-Java/app/build.gradle @@ -7,11 +7,11 @@ apply { } android { - namespace "com.tokbox.sample.screensharing" + namespace "com.tokbox.sample.devicescreensharing" compileSdkVersion extCompileSdkVersion defaultConfig { - applicationId "com.tokbox.sample.screensharing" + applicationId "com.tokbox.sample.devicescreensharing" minSdkVersion extMinSdkVersion targetSdkVersion extTargetSdkVersion versionCode extVersionCode diff --git a/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml b/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml index b78e7e5f..21a36a29 100644 --- a/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml +++ b/Device-Screen-Sharing-Java/app/src/main/AndroidManifest.xml @@ -1,6 +1,6 @@ + package="com.tokbox.sample.devicescreensharing"> diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java similarity index 99% rename from Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java rename to Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java index f10c16ec..4ff2a480 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.devicescreensharing; import android.Manifest; import android.content.Context; diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java similarity index 95% rename from Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java rename to Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java index 0af57414..03babc0a 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.devicescreensharing; import android.text.TextUtils; import androidx.annotation.NonNull; diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java similarity index 99% rename from Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java rename to Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java index 88a8ce76..04487294 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.devicescreensharing; import android.annotation.SuppressLint; import android.content.Context; diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingManager.java similarity index 97% rename from Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java rename to Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingManager.java index efd22409..e836370b 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingManager.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingManager.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.devicescreensharing; import android.annotation.TargetApi; import android.content.ComponentName; diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingService.java similarity index 98% rename from Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java rename to Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingService.java index 26ab69ff..6fde105d 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingService.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingService.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.devicescreensharing; import android.annotation.TargetApi; import android.app.Notification; From 617f608a65626b398414281555aff5e1ce73d5ec Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Wed, 1 Oct 2025 12:42:30 +0200 Subject: [PATCH 4/9] Rename projects and folders --- ... => build-webview-screen-sharing-java.yml} | 4 +- ...> build-webview-screen-sharing-kotlin.yml} | 4 +- Device-Screen-Sharing-Java/README.md | 124 +++++++----------- .../devicescreensharing/MainActivity.java | 11 -- .../devicescreensharing/OpenTokConfig.java | 2 +- .../app/src/main/res/values/strings.xml | 2 +- README.md | 3 +- .../.gitignore | 0 .../README.md | 2 +- .../app/.gitignore | 0 .../app/build.gradle | 4 +- .../app/src/main/AndroidManifest.xml | 2 +- .../webviewscreensharing}/MainActivity.java | 2 +- .../webviewscreensharing}/OpenTokConfig.java | 2 +- .../ScreenSharingCapturer.java | 2 +- .../app/src/main/res/layout/activity_main.xml | 0 .../src/main/res/mipmap-hdpi/ic_launcher.png | Bin .../src/main/res/mipmap-mdpi/ic_launcher.png | Bin .../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin .../main/res/mipmap-xxhdpi/ic_launcher.png | Bin .../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin .../app/src/main/res/values-w820dp/dimens.xml | 0 .../app/src/main/res/values/colors.xml | 0 .../app/src/main/res/values/dimens.xml | 0 .../app/src/main/res/values/strings.xml | 2 +- .../app/src/main/res/values/styles.xml | 0 .../build.gradle | 0 .../gradle.properties | 0 .../gradle/wrapper/gradle-wrapper.jar | Bin .../gradle/wrapper/gradle-wrapper.properties | 0 .../gradlew | 0 .../gradlew.bat | 0 .../settings.gradle | 0 .../.gitignore | 0 .../README.md | 4 +- .../app/.gitignore | 0 .../app/build.gradle | 4 +- .../app/src/main/AndroidManifest.xml | 2 +- .../webviewscreensharing}/MainActivity.kt | 2 +- .../webviewscreensharing}/OpenTokConfig.kt | 2 +- .../ScreenSharingCapturer.kt | 2 +- .../app/src/main/res/layout/activity_main.xml | 0 .../src/main/res/mipmap-hdpi/ic_launcher.png | Bin .../src/main/res/mipmap-mdpi/ic_launcher.png | Bin .../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin .../main/res/mipmap-xxhdpi/ic_launcher.png | Bin .../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin .../app/src/main/res/values-w820dp/dimens.xml | 0 .../app/src/main/res/values/colors.xml | 0 .../app/src/main/res/values/dimens.xml | 0 .../app/src/main/res/values/strings.xml | 2 +- .../app/src/main/res/values/styles.xml | 0 .../build.gradle | 0 .../gradle.properties | 0 .../gradle/wrapper/gradle-wrapper.jar | Bin .../gradle/wrapper/gradle-wrapper.properties | 0 .../gradlew | 0 .../gradlew.bat | 0 .../settings.gradle | 0 59 files changed, 75 insertions(+), 109 deletions(-) rename .github/workflows/{build-screen-sharing-java.yml => build-webview-screen-sharing-java.yml} (77%) rename .github/workflows/{build-screen-sharing-kotlin.yml => build-webview-screen-sharing-kotlin.yml} (77%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/.gitignore (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/README.md (99%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/.gitignore (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/build.gradle (89%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/AndroidManifest.xml (94%) rename {Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing => Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing}/MainActivity.java (99%) rename {Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing => Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing}/OpenTokConfig.java (95%) rename {Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing => Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing}/ScreenSharingCapturer.java (98%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/layout/activity_main.xml (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/mipmap-hdpi/ic_launcher.png (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/mipmap-mdpi/ic_launcher.png (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/mipmap-xhdpi/ic_launcher.png (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/mipmap-xxhdpi/ic_launcher.png (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/values-w820dp/dimens.xml (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/values/colors.xml (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/values/dimens.xml (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/values/strings.xml (71%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/app/src/main/res/values/styles.xml (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/build.gradle (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/gradle.properties (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/gradle/wrapper/gradle-wrapper.jar (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/gradle/wrapper/gradle-wrapper.properties (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/gradlew (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/gradlew.bat (100%) rename {Screen-Sharing-Java => Webview-Screen-Sharing-Java}/settings.gradle (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/.gitignore (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/README.md (99%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/.gitignore (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/build.gradle (92%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/AndroidManifest.xml (95%) rename {Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing => Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing}/MainActivity.kt (99%) rename {Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing => Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing}/OpenTokConfig.kt (96%) rename {Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing => Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing}/ScreenSharingCapturer.kt (98%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/layout/activity_main.xml (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/mipmap-hdpi/ic_launcher.png (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/mipmap-mdpi/ic_launcher.png (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/mipmap-xhdpi/ic_launcher.png (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/mipmap-xxhdpi/ic_launcher.png (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/values-w820dp/dimens.xml (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/values/colors.xml (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/values/dimens.xml (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/values/strings.xml (83%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/app/src/main/res/values/styles.xml (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/build.gradle (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/gradle.properties (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/gradle/wrapper/gradle-wrapper.jar (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/gradle/wrapper/gradle-wrapper.properties (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/gradlew (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/gradlew.bat (100%) rename {Screen-Sharing-Kotlin => Webview-Screen-Sharing-Kotlin}/settings.gradle (100%) diff --git a/.github/workflows/build-screen-sharing-java.yml b/.github/workflows/build-webview-screen-sharing-java.yml similarity index 77% rename from .github/workflows/build-screen-sharing-java.yml rename to .github/workflows/build-webview-screen-sharing-java.yml index 71fd11c9..309dc5d3 100644 --- a/.github/workflows/build-screen-sharing-java.yml +++ b/.github/workflows/build-webview-screen-sharing-java.yml @@ -1,4 +1,4 @@ -name: Build Screen-Sharing-Java +name: Build Webview-Screen-Sharing-Java on: push: @@ -22,4 +22,4 @@ jobs: java-version: 17 - name: Build - run: cd Screen-Sharing-Java && ./gradlew app:assembleRelease && cd .. + run: cd Webview-Screen-Sharing-Java && ./gradlew app:assembleRelease && cd .. diff --git a/.github/workflows/build-screen-sharing-kotlin.yml b/.github/workflows/build-webview-screen-sharing-kotlin.yml similarity index 77% rename from .github/workflows/build-screen-sharing-kotlin.yml rename to .github/workflows/build-webview-screen-sharing-kotlin.yml index 9e2a1cac..b24bbbc6 100644 --- a/.github/workflows/build-screen-sharing-kotlin.yml +++ b/.github/workflows/build-webview-screen-sharing-kotlin.yml @@ -1,4 +1,4 @@ -name: Build Screen-Sharing-Kotlin +name: Build Webview-Screen-Sharing-Kotlin on: push: @@ -22,4 +22,4 @@ jobs: java-version: 17 - name: Build - run: cd Screen-Sharing-Kotlin && ./gradlew app:assembleRelease && cd .. + run: cd Webview-Screen-Sharing-Kotlin && ./gradlew app:assembleRelease && cd .. diff --git a/Device-Screen-Sharing-Java/README.md b/Device-Screen-Sharing-Java/README.md index a2cd565c..84eb4ef6 100644 --- a/Device-Screen-Sharing-Java/README.md +++ b/Device-Screen-Sharing-Java/README.md @@ -1,27 +1,26 @@ # Screen Sharing -This app shows how to use `WebView` as the source for screen-sharing video. +This app demonstrates how to use the Media Projection API as the source for screen-sharing video. > Check [Basic-Video-Capturer-Camera-2](../Basic-Video-Capturer-Camera-2) project to see how a device camera can be used as the video source for the custom `Capturer`. + ## Screen sharing -Custom video capturer is using `WebView` from the Android application as the source of -a published stream. +The custom video capturer uses the Media Projection API to capture the device's screen and publish it as a video stream. -When the app starts up, the `onCreate` method instantiates a `WebView` object: +When the app starts up, the `onCreate` method initializes the Media Projection API by requesting permission to capture the screen: ```java -webViewContainer = findViewById(R.id.webview); +mediaProjectionManager = (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE); +startActivityForResult(mediaProjectionManager.createScreenCaptureIntent(), REQUEST_CODE); ``` -Upon connecting to the OpenTok session, the app instantiates a `Publisher` object, and calls its -`setCapturer` method to use a custom video capturer, defined by the `ScreenSharingCapturer` -class: +Upon connecting to the OpenTok session, the app instantiates a `Publisher` object and calls its `setCapturer` method to use a custom video capturer, defined by the `ScreenSharingCapturer` class: ```java @Override public void onConnected(Session session) { - ScreenSharingCapturer screenSharingCapturer = new ScreenSharingCapturer(MainActivity.this, webViewContainer); + ScreenSharingCapturer screenSharingCapturer = new ScreenSharingCapturer(MainActivity.this, mediaProjection); publisher = new Publisher.Builder(MainActivity.this) .capturer(screenSharingCapturer) @@ -31,12 +30,6 @@ public void onConnected(Session session) { publisher.setPublisherVideoType(PublisherKit.PublisherKitVideoType.PublisherKitVideoTypeScreen); publisher.setAudioFallbackEnabled(false); - webViewContainer.setWebViewClient(new WebViewClient()); - WebSettings webSettings = webViewContainer.getSettings(); - webSettings.setJavaScriptEnabled(true); - webViewContainer.setLayerType(View.LAYER_TYPE_SOFTWARE, null); - webViewContainer.loadUrl("https://www.tokbox.com"); - publisher.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL); publisherViewContainer.addView(publisher.getView()); @@ -44,21 +37,9 @@ public void onConnected(Session session) { } ``` -> Note: that the call to the `setPublisherVideoType` method sets the video type of the published -stream to `PublisherKitVideoType.PublisherKitVideoTypeScreen`. This optimizes the video encoding for -screen sharing. It is recommended to use a low frame rate (15 frames per second or lower) with this -video type. When using the screen video type in a session that uses the [OpenTok Media -Server](https://tokbox.com/opentok/tutorials/create-session/#media-mode), the -audio-only fallback feature is disabled, so that the video does not drop out in subscribers. - -The `onConnected` method also calls the `loadScreenWebView` method. This method -configures the WebView object, loading the TokBox URL. - -Note that the `webViewContainer` object is passed into the `ScreenSharingCapturer` constructor, -which assigns it to the `contentView` property. +> Note: The call to the `setPublisherVideoType` method sets the video type of the published stream to `PublisherKitVideoType.PublisherKitVideoTypeScreen`. This optimizes the video encoding for screen sharing. It is recommended to use a low frame rate (15 frames per second or lower) with this video type. When using the screen video type in a session that uses the [OpenTok Media Server](https://tokbox.com/opentok/tutorials/create-session/#media-mode), the audio-only fallback feature is disabled, so that the video does not drop out in subscribers. -The `getCaptureSettings` method initializes capture settings to be used by the custom -video capturer: +The `ScreenSharingCapturer` class uses the Media Projection API to capture the screen. The `getCaptureSettings` method initializes capture settings to be used by the custom video capturer: ```java @Override @@ -73,69 +54,64 @@ public CaptureSettings getCaptureSettings() { } ``` -The `startCapture` method starts the `frameProducer` thread after 1/15 second: +The `startCapture` method starts the screen capture process: ```java @Override public int startCapture() { capturing = true; + virtualDisplay = mediaProjection.createVirtualDisplay( + "ScreenSharing", + width, + height, + density, + DisplayManager.VIRTUAL_DISPLAY_FLAG_PUBLIC, + surface, + null, + null + ); + handler.postDelayed(newFrame, 1000 / fps); return 0; } ``` -The `frameProducer` thread gets a `Bitmap` representation of the `contentView` object - (the `WebView`), writes its pixels to a buffer, and then calls the `provideIntArrayFrame()` - method, passing in that buffer as a parameter: +The `backgroundHandler` thread captures frames from the virtual display, processes them, and sends them to the publisher: ```java -private Runnable newFrame = new Runnable() { - @Override - public void run() { - if (capturing) { - int width = contentView.getWidth(); - int height = contentView.getHeight(); - - if (frame == null || - ScreenSharingCapturer.this.width != width || - ScreenSharingCapturer.this.height != height) { - - ScreenSharingCapturer.this.width = width; - ScreenSharingCapturer.this.height = height; - - if (bmp != null) { - bmp.recycle(); - bmp = null; +imageReader.setOnImageAvailableListener(new ImageReader.OnImageAvailableListener() { + @Override + public void onImageAvailable(ImageReader reader) { + Image image = reader.acquireLatestImage(); + if (image != null) { + Image.Plane[] planes = image.getPlanes(); + ByteBuffer buffer = planes[0].getBuffer(); + int pixelStride = planes[0].getPixelStride(); + int rowStride = planes[0].getRowStride(); + + if (frame == null) { + frame = new int[width * height]; + } + + for (int y = 0; y < height; y++) { + for (int x = 0; x < width; x++) { + int index = y * rowStride + x * pixelStride; + int pixel = buffer.getInt(index); + frame[y * width + x] = pixel; + } + } + + provideIntArrayFrame(frame, ABGR, width, height, 0, false); + image.close(); } - - bmp = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); - - canvas = new Canvas(bmp); - frame = new int[width * height]; } - canvas.saveLayer(0, 0, width, height, null); - canvas.translate(-contentView.getScrollX(), - contentView.getScrollY()); - contentView.draw(canvas); - - bmp.getPixels(frame, 0, width, 0, 0, width, height); - - provideIntArrayFrame(frame, ARGB, width, height, 0, false); - - canvas.restore(); - - handler.postDelayed(newFrame, 1000 / fps); - - } - } -}; + }, backgroundHandler); ``` -The `provideIntArrayFrame` method, defined by the `BaseVideoCapturer` class sends an integer array of data to the publisher, to be used for the next video frame published. +The `provideIntArrayFrame` method, defined by the `BaseVideoCapturer` class, sends an integer array of data to the publisher, to be used for the next video frame published. -If the publisher is still capturing video, the thread starts again after another 1/15 of a -second, so that the capturer continues to supply the publisher with new video frames to publish. -``` +If the publisher is still capturing video, the thread starts again after another 1/15 of a second, so that the capturer continues to supply the publisher with new video frames to publish. ## Further Reading diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java index 4ff2a480..dd14ae7e 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java @@ -9,14 +9,8 @@ import android.os.Build; import android.os.Bundle; import android.os.Handler; -import android.provider.MediaStore; import android.util.Log; -import android.view.View; -import android.webkit.WebSettings; -import android.webkit.WebView; -import android.webkit.WebViewClient; import android.widget.FrameLayout; -import android.widget.RelativeLayout; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.appcompat.app.AppCompatActivity; @@ -32,7 +26,6 @@ import pub.devrel.easypermissions.AfterPermissionGranted; import pub.devrel.easypermissions.EasyPermissions; import java.util.List; -import java.util.Objects; public class MainActivity extends AppCompatActivity implements EasyPermissions.PermissionCallbacks { @@ -242,10 +235,6 @@ public void onRequestPermissionsResult(int requestCode, @NonNull String[] permis @Override public void onPermissionsGranted(int requestCode, List perms) { Log.d(TAG, "onPermissionsGranted:" + requestCode + ": " + perms); - - if (requestCode == PERMISSIONS_REQUEST_CODE) { - initializeSession(OpenTokConfig.API_KEY, OpenTokConfig.SESSION_ID, OpenTokConfig.TOKEN); - } } @Override diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java index 03babc0a..0af57414 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.devicescreensharing; +package com.tokbox.sample.screensharing; import android.text.TextUtils; import androidx.annotation.NonNull; diff --git a/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml b/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml index 7367ce56..e769538c 100644 --- a/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml +++ b/Device-Screen-Sharing-Java/app/src/main/res/values/strings.xml @@ -1,4 +1,4 @@ - Screen-Sharing + Device-Screen-Sharing This app needs access to your camera and mic so you can perform video calls diff --git a/README.md b/README.md index fcbb911b..c7900f87 100644 --- a/README.md +++ b/README.md @@ -18,9 +18,10 @@ The Android projects in this directory demonstrate typical use cases and feature - Basic-Audio-Driver ([Java](./Basic-Audio-Driver-Java), [Kotlin](./Basic-Audio-Driver-Kotlin)) demonstrates how to publish a random audio signal and save audio streams to the file. - Advanced-Audio-Driver ([Java](./Advanced-Audio-Driver-Java), [Kotlin](./Advanced-Audio-Driver-Kotlin)) demonstrates how to create a more advanced custom audio driver - Basic-Video-Driver ([Java](./Basic-Video-Driver-Java)) demonstrates how to create a custom video driver +- Device-Screen-Sharing ([Java](./Webview-Screen-Sharing-Java)) demonstrates how to publish a screen-sharing video, using the entire device screen as the source - Live-Photo-Capture ([Java](./Live-Photo-Capture-Java), [Kotlin](./Live-Photo-Capture-Kotlin)) demonstrates how to capture an image from a subscribed video stream - Picture-In-Picture ([Java](./Picture-In-Picture-Java)) demonstrates how to use the [Picture-in-Picture](https://developer.android.com/guide/topics/ui/picture-in-picture) mode -- Screen-Sharing ([Java](./Screen-Sharing-Java), [Kotlin](./Screen-Sharing-Kotlin)) demonstrates how to publish a screen-sharing video, using the WebView as the source +- Webview-Screen-Sharing ([Java](./Webview-Screen-Sharing-Java), [Kotlin](./Webview-Screen-Sharing-Kotlin)) demonstrates how to publish a webview-screen-sharing video, using the WebView as the source - Phone-Call-Detection ([Java](./Phone-Call-Detection-Java), [Kotlin](./Phone-Call-Detection-Kotlin)) demonstrates how to detect incoming and outgoing phone calls - ARCore-Integration ([Java](./ARCore-Integration-Java)) demonstrates how to use Google [ARCore](https://developers.google.com/ar) with Opentok - Basic-VoIP-Call ([Java](./Basic-VoIP-Call-Java)) demonstrates how to use Android Connection Service (https://developer.android.com/reference/android/telecom/ConnectionService) with the OpenTok Android SDK. diff --git a/Screen-Sharing-Java/.gitignore b/Webview-Screen-Sharing-Java/.gitignore similarity index 100% rename from Screen-Sharing-Java/.gitignore rename to Webview-Screen-Sharing-Java/.gitignore diff --git a/Screen-Sharing-Java/README.md b/Webview-Screen-Sharing-Java/README.md similarity index 99% rename from Screen-Sharing-Java/README.md rename to Webview-Screen-Sharing-Java/README.md index a2cd565c..dbec97ef 100644 --- a/Screen-Sharing-Java/README.md +++ b/Webview-Screen-Sharing-Java/README.md @@ -1,4 +1,4 @@ -# Screen Sharing +# Webview Screen Sharing This app shows how to use `WebView` as the source for screen-sharing video. diff --git a/Screen-Sharing-Java/app/.gitignore b/Webview-Screen-Sharing-Java/app/.gitignore similarity index 100% rename from Screen-Sharing-Java/app/.gitignore rename to Webview-Screen-Sharing-Java/app/.gitignore diff --git a/Screen-Sharing-Java/app/build.gradle b/Webview-Screen-Sharing-Java/app/build.gradle similarity index 89% rename from Screen-Sharing-Java/app/build.gradle rename to Webview-Screen-Sharing-Java/app/build.gradle index b4d37d7b..a3236a01 100644 --- a/Screen-Sharing-Java/app/build.gradle +++ b/Webview-Screen-Sharing-Java/app/build.gradle @@ -7,11 +7,11 @@ apply { } android { - namespace "com.tokbox.sample.screensharing" + namespace "com.tokbox.sample.webviewscreensharing" compileSdkVersion extCompileSdkVersion defaultConfig { - applicationId "com.tokbox.sample.screensharing" + applicationId "com.tokbox.sample.webviewscreensharing" minSdkVersion extMinSdkVersion targetSdkVersion extTargetSdkVersion versionCode extVersionCode diff --git a/Screen-Sharing-Java/app/src/main/AndroidManifest.xml b/Webview-Screen-Sharing-Java/app/src/main/AndroidManifest.xml similarity index 94% rename from Screen-Sharing-Java/app/src/main/AndroidManifest.xml rename to Webview-Screen-Sharing-Java/app/src/main/AndroidManifest.xml index 63a78a4d..4ee3ba77 100644 --- a/Screen-Sharing-Java/app/src/main/AndroidManifest.xml +++ b/Webview-Screen-Sharing-Java/app/src/main/AndroidManifest.xml @@ -1,6 +1,6 @@ + package="com.tokbox.sample.webviewscreensharing"> diff --git a/Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java b/Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/MainActivity.java similarity index 99% rename from Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java rename to Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/MainActivity.java index 2e9c9752..41a32131 100644 --- a/Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.java +++ b/Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/MainActivity.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.webviewscreensharing; import android.Manifest; import android.os.Bundle; diff --git a/Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java b/Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/OpenTokConfig.java similarity index 95% rename from Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java rename to Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/OpenTokConfig.java index 0af57414..f23c56b2 100644 --- a/Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.java +++ b/Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/OpenTokConfig.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.webviewscreensharing; import android.text.TextUtils; import androidx.annotation.NonNull; diff --git a/Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java b/Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/ScreenSharingCapturer.java similarity index 98% rename from Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java rename to Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/ScreenSharingCapturer.java index 1578bd94..e53bcec5 100644 --- a/Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.java +++ b/Webview-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/webviewscreensharing/ScreenSharingCapturer.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.webviewscreensharing; import android.content.Context; import android.graphics.Bitmap; diff --git a/Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml b/Webview-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml rename to Webview-Screen-Sharing-Java/app/src/main/res/layout/activity_main.xml diff --git a/Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png b/Webview-Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png rename to Webview-Screen-Sharing-Java/app/src/main/res/mipmap-hdpi/ic_launcher.png diff --git a/Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png b/Webview-Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png rename to Webview-Screen-Sharing-Java/app/src/main/res/mipmap-mdpi/ic_launcher.png diff --git a/Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/Webview-Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png rename to Webview-Screen-Sharing-Java/app/src/main/res/mipmap-xhdpi/ic_launcher.png diff --git a/Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/Webview-Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png rename to Webview-Screen-Sharing-Java/app/src/main/res/mipmap-xxhdpi/ic_launcher.png diff --git a/Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/Webview-Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png rename to Webview-Screen-Sharing-Java/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png diff --git a/Screen-Sharing-Java/app/src/main/res/values-w820dp/dimens.xml b/Webview-Screen-Sharing-Java/app/src/main/res/values-w820dp/dimens.xml similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/values-w820dp/dimens.xml rename to Webview-Screen-Sharing-Java/app/src/main/res/values-w820dp/dimens.xml diff --git a/Screen-Sharing-Java/app/src/main/res/values/colors.xml b/Webview-Screen-Sharing-Java/app/src/main/res/values/colors.xml similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/values/colors.xml rename to Webview-Screen-Sharing-Java/app/src/main/res/values/colors.xml diff --git a/Screen-Sharing-Java/app/src/main/res/values/dimens.xml b/Webview-Screen-Sharing-Java/app/src/main/res/values/dimens.xml similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/values/dimens.xml rename to Webview-Screen-Sharing-Java/app/src/main/res/values/dimens.xml diff --git a/Screen-Sharing-Java/app/src/main/res/values/strings.xml b/Webview-Screen-Sharing-Java/app/src/main/res/values/strings.xml similarity index 71% rename from Screen-Sharing-Java/app/src/main/res/values/strings.xml rename to Webview-Screen-Sharing-Java/app/src/main/res/values/strings.xml index 7367ce56..729df24a 100644 --- a/Screen-Sharing-Java/app/src/main/res/values/strings.xml +++ b/Webview-Screen-Sharing-Java/app/src/main/res/values/strings.xml @@ -1,4 +1,4 @@ - Screen-Sharing + Webview-Screen-Sharing This app needs access to your camera and mic so you can perform video calls diff --git a/Screen-Sharing-Java/app/src/main/res/values/styles.xml b/Webview-Screen-Sharing-Java/app/src/main/res/values/styles.xml similarity index 100% rename from Screen-Sharing-Java/app/src/main/res/values/styles.xml rename to Webview-Screen-Sharing-Java/app/src/main/res/values/styles.xml diff --git a/Screen-Sharing-Java/build.gradle b/Webview-Screen-Sharing-Java/build.gradle similarity index 100% rename from Screen-Sharing-Java/build.gradle rename to Webview-Screen-Sharing-Java/build.gradle diff --git a/Screen-Sharing-Java/gradle.properties b/Webview-Screen-Sharing-Java/gradle.properties similarity index 100% rename from Screen-Sharing-Java/gradle.properties rename to Webview-Screen-Sharing-Java/gradle.properties diff --git a/Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar b/Webview-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar similarity index 100% rename from Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar rename to Webview-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.jar diff --git a/Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties b/Webview-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties similarity index 100% rename from Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties rename to Webview-Screen-Sharing-Java/gradle/wrapper/gradle-wrapper.properties diff --git a/Screen-Sharing-Java/gradlew b/Webview-Screen-Sharing-Java/gradlew similarity index 100% rename from Screen-Sharing-Java/gradlew rename to Webview-Screen-Sharing-Java/gradlew diff --git a/Screen-Sharing-Java/gradlew.bat b/Webview-Screen-Sharing-Java/gradlew.bat similarity index 100% rename from Screen-Sharing-Java/gradlew.bat rename to Webview-Screen-Sharing-Java/gradlew.bat diff --git a/Screen-Sharing-Java/settings.gradle b/Webview-Screen-Sharing-Java/settings.gradle similarity index 100% rename from Screen-Sharing-Java/settings.gradle rename to Webview-Screen-Sharing-Java/settings.gradle diff --git a/Screen-Sharing-Kotlin/.gitignore b/Webview-Screen-Sharing-Kotlin/.gitignore similarity index 100% rename from Screen-Sharing-Kotlin/.gitignore rename to Webview-Screen-Sharing-Kotlin/.gitignore diff --git a/Screen-Sharing-Kotlin/README.md b/Webview-Screen-Sharing-Kotlin/README.md similarity index 99% rename from Screen-Sharing-Kotlin/README.md rename to Webview-Screen-Sharing-Kotlin/README.md index 5a2f3306..dae36367 100644 --- a/Screen-Sharing-Kotlin/README.md +++ b/Webview-Screen-Sharing-Kotlin/README.md @@ -1,9 +1,9 @@ -# Screen Sharing +# Webview Screen Sharing This app shows how to use `WebView` as the source for screen-sharing video. > Check [Basic-Video-Capturer-Camera-2](../Basic-Video-Capturer-Camera-2) project to see how a device camera can be used as the video source for the custom `Capturer`. -## Screen sharing +## Webview Screen sharing Custom video capturer is using `WebView` from the Android application as the source of a published stream. diff --git a/Screen-Sharing-Kotlin/app/.gitignore b/Webview-Screen-Sharing-Kotlin/app/.gitignore similarity index 100% rename from Screen-Sharing-Kotlin/app/.gitignore rename to Webview-Screen-Sharing-Kotlin/app/.gitignore diff --git a/Screen-Sharing-Kotlin/app/build.gradle b/Webview-Screen-Sharing-Kotlin/app/build.gradle similarity index 92% rename from Screen-Sharing-Kotlin/app/build.gradle rename to Webview-Screen-Sharing-Kotlin/app/build.gradle index 7f155b6e..12b49037 100644 --- a/Screen-Sharing-Kotlin/app/build.gradle +++ b/Webview-Screen-Sharing-Kotlin/app/build.gradle @@ -8,11 +8,11 @@ apply { } android { - namespace "com.tokbox.sample.screensharing" + namespace "com.tokbox.sample.webviewscreensharing" compileSdkVersion extCompileSdkVersion defaultConfig { - applicationId "com.tokbox.sample.screensharing" + applicationId "com.tokbox.sample.webviewscreensharing" minSdkVersion extMinSdkVersion targetSdkVersion extTargetSdkVersion versionCode extVersionCode diff --git a/Screen-Sharing-Kotlin/app/src/main/AndroidManifest.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/AndroidManifest.xml similarity index 95% rename from Screen-Sharing-Kotlin/app/src/main/AndroidManifest.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/AndroidManifest.xml index 9b370111..0c75d1cb 100644 --- a/Screen-Sharing-Kotlin/app/src/main/AndroidManifest.xml +++ b/Webview-Screen-Sharing-Kotlin/app/src/main/AndroidManifest.xml @@ -1,6 +1,6 @@ + package="com.tokbox.sample.webviewscreensharing" > diff --git a/Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.kt b/Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/MainActivity.kt similarity index 99% rename from Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.kt rename to Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/MainActivity.kt index 65092c6e..a013c06d 100644 --- a/Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/MainActivity.kt +++ b/Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/MainActivity.kt @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing +package com.tokbox.sample.webviewscreensharing import android.Manifest import android.opengl.GLSurfaceView diff --git a/Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.kt b/Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/OpenTokConfig.kt similarity index 96% rename from Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.kt rename to Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/OpenTokConfig.kt index ed46ebb6..21a05309 100644 --- a/Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/OpenTokConfig.kt +++ b/Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/OpenTokConfig.kt @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing +package com.tokbox.sample.webviewscreensharing import android.text.TextUtils diff --git a/Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.kt b/Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/ScreenSharingCapturer.kt similarity index 98% rename from Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.kt rename to Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/ScreenSharingCapturer.kt index bd754674..c8640fc2 100644 --- a/Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/screensharing/ScreenSharingCapturer.kt +++ b/Webview-Screen-Sharing-Kotlin/app/src/main/java/com/tokbox/sample/webviewscreensharing/ScreenSharingCapturer.kt @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing +package com.tokbox.sample.webviewscreensharing import android.content.Context import android.graphics.Bitmap diff --git a/Screen-Sharing-Kotlin/app/src/main/res/layout/activity_main.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/res/layout/activity_main.xml similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/layout/activity_main.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/layout/activity_main.xml diff --git a/Screen-Sharing-Kotlin/app/src/main/res/mipmap-hdpi/ic_launcher.png b/Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-hdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/mipmap-hdpi/ic_launcher.png rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-hdpi/ic_launcher.png diff --git a/Screen-Sharing-Kotlin/app/src/main/res/mipmap-mdpi/ic_launcher.png b/Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-mdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/mipmap-mdpi/ic_launcher.png rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-mdpi/ic_launcher.png diff --git a/Screen-Sharing-Kotlin/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-xhdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/mipmap-xhdpi/ic_launcher.png rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-xhdpi/ic_launcher.png diff --git a/Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxhdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxhdpi/ic_launcher.png rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxhdpi/ic_launcher.png diff --git a/Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png diff --git a/Screen-Sharing-Kotlin/app/src/main/res/values-w820dp/dimens.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/res/values-w820dp/dimens.xml similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/values-w820dp/dimens.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/values-w820dp/dimens.xml diff --git a/Screen-Sharing-Kotlin/app/src/main/res/values/colors.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/res/values/colors.xml similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/values/colors.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/values/colors.xml diff --git a/Screen-Sharing-Kotlin/app/src/main/res/values/dimens.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/res/values/dimens.xml similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/values/dimens.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/values/dimens.xml diff --git a/Screen-Sharing-Kotlin/app/src/main/res/values/strings.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/res/values/strings.xml similarity index 83% rename from Screen-Sharing-Kotlin/app/src/main/res/values/strings.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/values/strings.xml index 13f03a21..0e7429e8 100644 --- a/Screen-Sharing-Kotlin/app/src/main/res/values/strings.xml +++ b/Webview-Screen-Sharing-Kotlin/app/src/main/res/values/strings.xml @@ -1,7 +1,7 @@ - Screen-Sharing + Webview-Screen-Sharing Settings Permissions Required diff --git a/Screen-Sharing-Kotlin/app/src/main/res/values/styles.xml b/Webview-Screen-Sharing-Kotlin/app/src/main/res/values/styles.xml similarity index 100% rename from Screen-Sharing-Kotlin/app/src/main/res/values/styles.xml rename to Webview-Screen-Sharing-Kotlin/app/src/main/res/values/styles.xml diff --git a/Screen-Sharing-Kotlin/build.gradle b/Webview-Screen-Sharing-Kotlin/build.gradle similarity index 100% rename from Screen-Sharing-Kotlin/build.gradle rename to Webview-Screen-Sharing-Kotlin/build.gradle diff --git a/Screen-Sharing-Kotlin/gradle.properties b/Webview-Screen-Sharing-Kotlin/gradle.properties similarity index 100% rename from Screen-Sharing-Kotlin/gradle.properties rename to Webview-Screen-Sharing-Kotlin/gradle.properties diff --git a/Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.jar b/Webview-Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.jar similarity index 100% rename from Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.jar rename to Webview-Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.jar diff --git a/Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.properties b/Webview-Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.properties similarity index 100% rename from Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.properties rename to Webview-Screen-Sharing-Kotlin/gradle/wrapper/gradle-wrapper.properties diff --git a/Screen-Sharing-Kotlin/gradlew b/Webview-Screen-Sharing-Kotlin/gradlew similarity index 100% rename from Screen-Sharing-Kotlin/gradlew rename to Webview-Screen-Sharing-Kotlin/gradlew diff --git a/Screen-Sharing-Kotlin/gradlew.bat b/Webview-Screen-Sharing-Kotlin/gradlew.bat similarity index 100% rename from Screen-Sharing-Kotlin/gradlew.bat rename to Webview-Screen-Sharing-Kotlin/gradlew.bat diff --git a/Screen-Sharing-Kotlin/settings.gradle b/Webview-Screen-Sharing-Kotlin/settings.gradle similarity index 100% rename from Screen-Sharing-Kotlin/settings.gradle rename to Webview-Screen-Sharing-Kotlin/settings.gradle From baf5c5dc63fcbe373f87d71b213f7eaf123b228a Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Wed, 1 Oct 2025 12:49:02 +0200 Subject: [PATCH 5/9] Fix consitencies --- ...-sharing-java.yml => build-device-screen-sharing-java.yml} | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) rename .github/workflows/{build-basic-device-screen-sharing-java.yml => build-device-screen-sharing-java.yml} (76%) diff --git a/.github/workflows/build-basic-device-screen-sharing-java.yml b/.github/workflows/build-device-screen-sharing-java.yml similarity index 76% rename from .github/workflows/build-basic-device-screen-sharing-java.yml rename to .github/workflows/build-device-screen-sharing-java.yml index db265d6d..29ce9c2c 100644 --- a/.github/workflows/build-basic-device-screen-sharing-java.yml +++ b/.github/workflows/build-device-screen-sharing-java.yml @@ -1,4 +1,4 @@ -name: Build Basic-Device-Screen-Sharing-Java +name: Build Device-Screen-Sharing-Java on: push: @@ -22,4 +22,4 @@ jobs: java-version: 17 - name: Build - run: cd Basic-Device-Screen-Sharing-Java && ./gradlew app:assembleRelease && cd .. + run: cd Device-Screen-Sharing-Java && ./gradlew app:assembleRelease && cd .. From 61dd7ffe041f61a1e548911fdc5d94414832c7b5 Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Wed, 1 Oct 2025 12:51:08 +0200 Subject: [PATCH 6/9] Fix consitencies --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index c7900f87..ab362860 100644 --- a/README.md +++ b/README.md @@ -18,7 +18,7 @@ The Android projects in this directory demonstrate typical use cases and feature - Basic-Audio-Driver ([Java](./Basic-Audio-Driver-Java), [Kotlin](./Basic-Audio-Driver-Kotlin)) demonstrates how to publish a random audio signal and save audio streams to the file. - Advanced-Audio-Driver ([Java](./Advanced-Audio-Driver-Java), [Kotlin](./Advanced-Audio-Driver-Kotlin)) demonstrates how to create a more advanced custom audio driver - Basic-Video-Driver ([Java](./Basic-Video-Driver-Java)) demonstrates how to create a custom video driver -- Device-Screen-Sharing ([Java](./Webview-Screen-Sharing-Java)) demonstrates how to publish a screen-sharing video, using the entire device screen as the source +- Device-Screen-Sharing ([Java](./Device-Screen-Sharing-Java)) demonstrates how to publish a screen-sharing video, using the entire device screen as the source - Live-Photo-Capture ([Java](./Live-Photo-Capture-Java), [Kotlin](./Live-Photo-Capture-Kotlin)) demonstrates how to capture an image from a subscribed video stream - Picture-In-Picture ([Java](./Picture-In-Picture-Java)) demonstrates how to use the [Picture-in-Picture](https://developer.android.com/guide/topics/ui/picture-in-picture) mode - Webview-Screen-Sharing ([Java](./Webview-Screen-Sharing-Java), [Kotlin](./Webview-Screen-Sharing-Kotlin)) demonstrates how to publish a webview-screen-sharing video, using the WebView as the source From 7e2d32f239a945c35d85279d9bad7f04b00ee09e Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Wed, 1 Oct 2025 12:54:28 +0200 Subject: [PATCH 7/9] Fix building error --- .../com/tokbox/sample/devicescreensharing/OpenTokConfig.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java index 0af57414..03babc0a 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/OpenTokConfig.java @@ -1,4 +1,4 @@ -package com.tokbox.sample.screensharing; +package com.tokbox.sample.devicescreensharing; import android.text.TextUtils; import androidx.annotation.NonNull; From 74e447c5865b90bc068e81a436bd46b203729ecd Mon Sep 17 00:00:00 2001 From: Goncalo Mendes <106695857+goncalocostamendes@users.noreply.github.com> Date: Wed, 1 Oct 2025 13:13:09 +0200 Subject: [PATCH 8/9] Update Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> --- .../com/tokbox/sample/devicescreensharing/MainActivity.java | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java index dd14ae7e..0c1b7920 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java @@ -116,11 +116,7 @@ private void startScreenCapture(int resultCode, Intent data) { public void onConnected(Session session) { Log.d(TAG, "onConnected: Connected to session " + session.getSessionId()); - if (Build.VERSION.SDK_INT >= 34) { - requestScreenCapturePermission(); - } else { - requestScreenCapturePermission(); - } + requestScreenCapturePermission(); } @Override From 98a3bbb176a6500af3bf1e148db6e9a5c7fc379f Mon Sep 17 00:00:00 2001 From: Goncalo Mendes Date: Fri, 3 Oct 2025 11:11:43 +0200 Subject: [PATCH 9/9] Optimise screen sharing capturer and suppot api level below 34 --- .../sample/devicescreensharing/MainActivity.java | 16 ++++++++++++++-- .../ScreenSharingCapturer.java | 15 +-------------- 2 files changed, 15 insertions(+), 16 deletions(-) diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java index 0c1b7920..3c94c396 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/MainActivity.java @@ -26,6 +26,7 @@ import pub.devrel.easypermissions.AfterPermissionGranted; import pub.devrel.easypermissions.EasyPermissions; import java.util.List; +import java.util.ArrayList; public class MainActivity extends AppCompatActivity implements EasyPermissions.PermissionCallbacks { @@ -240,8 +241,19 @@ public void onPermissionsDenied(int requestCode, List perms) { @AfterPermissionGranted(PERMISSIONS_REQUEST_CODE) private void requestPermissions() { - String[] perms = {Manifest.permission.INTERNET, Manifest.permission.CAMERA, Manifest.permission.RECORD_AUDIO, Manifest.permission.FOREGROUND_SERVICE, Manifest.permission.FOREGROUND_SERVICE_MEDIA_PROJECTION}; - + ArrayList permsList = new ArrayList<>(); + permsList.add(Manifest.permission.INTERNET); + permsList.add(Manifest.permission.CAMERA); + permsList.add(Manifest.permission.RECORD_AUDIO); + permsList.add(Manifest.permission.FOREGROUND_SERVICE); + + // Add FOREGROUND_SERVICE_MEDIA_PROJECTION permission if API level is 34 or higher + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKE) { + permsList.add(Manifest.permission.FOREGROUND_SERVICE_MEDIA_PROJECTION); + } + + String[] perms = permsList.toArray(new String[0]); + if (EasyPermissions.hasPermissions(this, perms)) { initializeSession(OpenTokConfig.API_KEY, OpenTokConfig.SESSION_ID, OpenTokConfig.TOKEN); } else { diff --git a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java index 04487294..d93df5f7 100644 --- a/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java +++ b/Device-Screen-Sharing-Java/app/src/main/java/com/tokbox/sample/devicescreensharing/ScreenSharingCapturer.java @@ -39,7 +39,6 @@ public class ScreenSharingCapturer extends BaseVideoCapturer { private int fps = 15; private int width = 0; private int height = 0; - private int[] frame; public ScreenSharingCapturer(Context context, MediaProjection mediaProjection) { this.context = context; @@ -96,19 +95,7 @@ public void onImageAvailable(ImageReader reader) { int pixelStride = planes[0].getPixelStride(); int rowStride = planes[0].getRowStride(); - if (frame == null) { - frame = new int[width * height]; - } - - for (int y = 0; y < height; y++) { - for (int x = 0; x < width; x++) { - int index = y * rowStride + x * pixelStride; - int pixel = buffer.getInt(index); - frame[y * width + x] = pixel; - } - } - - provideIntArrayFrame(frame, ABGR, width, height, 0, false); + provideBufferFrame(planes[0].getBuffer(), ABGR, width, height, 0, false); image.close(); } }