WIP
This commit is contained in:
parent
7918bf39f5
commit
45519cdd27
7 changed files with 94 additions and 7 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -755,6 +755,7 @@ name = "capture"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"block",
|
"block",
|
||||||
|
"cc",
|
||||||
"cocoa",
|
"cocoa",
|
||||||
"core-foundation",
|
"core-foundation",
|
||||||
"core-graphics",
|
"core-graphics",
|
||||||
|
|
|
@ -19,3 +19,6 @@ foreign-types = "0.3"
|
||||||
log = { version = "0.4.16", features = ["kv_unstable_serde"] }
|
log = { version = "0.4.16", features = ["kv_unstable_serde"] }
|
||||||
objc = "0.2"
|
objc = "0.2"
|
||||||
simplelog = "0.9"
|
simplelog = "0.9"
|
||||||
|
|
||||||
|
[build-dependencies]
|
||||||
|
cc = "1.0"
|
|
@ -1,3 +1,10 @@
|
||||||
fn main() {
|
fn main() {
|
||||||
println!("cargo:rustc-link-lib=framework=ScreenCaptureKit");
|
println!("cargo:rustc-link-lib=framework=ScreenCaptureKit");
|
||||||
|
println!("cargo:rustc-link-lib=framework=CoreMedia");
|
||||||
|
println!("cargo:rustc-env=MACOSX_DEPLOYMENT_TARGET=12.3");
|
||||||
|
|
||||||
|
cc::Build::new()
|
||||||
|
.file("src/dummy.m")
|
||||||
|
.define("MACOSX_DEPLOYMENT_TARGET", "12.3")
|
||||||
|
.compile("dummy");
|
||||||
}
|
}
|
||||||
|
|
12
crates/capture/src/dummy.m
Normal file
12
crates/capture/src/dummy.m
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
#import <ScreenCaptureKit/ScreenCaptureKit.h>
|
||||||
|
|
||||||
|
@interface MyClass : NSObject <SCStreamOutput>
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation MyClass
|
||||||
|
- (void)stream:(SCStream *)stream
|
||||||
|
didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer
|
||||||
|
ofType:(SCStreamOutputType)type {
|
||||||
|
}
|
||||||
|
|
||||||
|
@end
|
|
@ -1,13 +1,14 @@
|
||||||
use std::{slice, str};
|
use std::{ffi::CStr, slice, str};
|
||||||
|
|
||||||
use block::ConcreteBlock;
|
use block::ConcreteBlock;
|
||||||
use cocoa::{
|
use cocoa::{
|
||||||
base::id,
|
base::{id, nil},
|
||||||
foundation::{NSString, NSUInteger},
|
foundation::{NSArray, NSString, NSUInteger, NSInteger},
|
||||||
};
|
};
|
||||||
use gpui::{actions, elements::*, keymap::Binding, Menu, MenuItem};
|
use core_graphics::display::CGDirectDisplayID;
|
||||||
|
use gpui::{actions, elements::*, keymap::Binding, Menu, MenuItem, mac::dispatcher::dispatch_get_main_queue};
|
||||||
use log::LevelFilter;
|
use log::LevelFilter;
|
||||||
use objc::{class, msg_send, sel, sel_impl};
|
use objc::{class, msg_send, sel, sel_impl, declare::ClassDecl, runtime::{Protocol, Object, Sel}};
|
||||||
use simplelog::SimpleLogger;
|
use simplelog::SimpleLogger;
|
||||||
|
|
||||||
#[allow(non_upper_case_globals)]
|
#[allow(non_upper_case_globals)]
|
||||||
|
@ -32,13 +33,57 @@ fn main() {
|
||||||
}]);
|
}]);
|
||||||
|
|
||||||
unsafe {
|
unsafe {
|
||||||
|
|
||||||
|
|
||||||
let block = ConcreteBlock::new(move |content: id, error: id| {
|
let block = ConcreteBlock::new(move |content: id, error: id| {
|
||||||
println!(
|
println!(
|
||||||
"got response with shareable content {:?} {:?} {:?}",
|
"got response with shareable content {:?} {:?} {:?}",
|
||||||
content,
|
content,
|
||||||
error,
|
error,
|
||||||
string_from_objc(msg_send![error, localizedDescription]),
|
string_from_objc(msg_send![error, localizedDescription]),
|
||||||
)
|
);
|
||||||
|
|
||||||
|
let displays: id = msg_send![content, displays];
|
||||||
|
if let Some(display) = (0..displays.count())
|
||||||
|
.map(|ix| displays.objectAtIndex(ix))
|
||||||
|
.next()
|
||||||
|
{
|
||||||
|
let mut decl = ClassDecl::new("CaptureOutput", class!(NSObject)).unwrap();
|
||||||
|
decl.add_protocol(Protocol::get("SCStreamOutput").unwrap());
|
||||||
|
decl.add_method(sel!(stream:didOutputSampleBuffer:ofType:), sample_output as extern "C" fn(&Object, Sel, id, id, NSInteger));
|
||||||
|
let capture_output_class = decl.register();
|
||||||
|
|
||||||
|
let output: id = msg_send![capture_output_class, alloc];
|
||||||
|
let output: id = msg_send![output, init];
|
||||||
|
|
||||||
|
let excluded_windows: id = msg_send![class!(NSArray), array];
|
||||||
|
let filter: id = msg_send![class!(SCContentFilter), alloc];
|
||||||
|
let filter: id = msg_send![filter, initWithDisplay: display excludingWindows: excluded_windows];
|
||||||
|
let config: id = msg_send![class!(SCStreamConfiguration), alloc];
|
||||||
|
// Configure the display content width and height.
|
||||||
|
let _: () = msg_send![config, setWidth: 800];
|
||||||
|
let _: () = msg_send![config, setHeight: 600];
|
||||||
|
let _: () = msg_send![config, setMinimumFrameInterval: CMTimeMake(1, 60)];
|
||||||
|
let _: () = msg_send![config, setQueueDepth: 5];
|
||||||
|
|
||||||
|
let stream: id = msg_send![class!(SCStream), alloc];
|
||||||
|
let stream: id = msg_send![stream, initWithFilter: filter configuration: config delegate: nil];
|
||||||
|
let error: id = nil;
|
||||||
|
let _: () = msg_send![stream, addStreamOutput: output type: 0 sampleHandlerQueue: dispatch_get_main_queue() error: &error];
|
||||||
|
println!("Added stream output... error? {}", string_from_objc(msg_send![error, localizedDescription]));
|
||||||
|
|
||||||
|
|
||||||
|
let start_capture_completion = ConcreteBlock::new(move |error: id| {
|
||||||
|
println!("Started capturing... error? {}", string_from_objc(msg_send![error, localizedDescription]));
|
||||||
|
println!("recovery suggestion {}", string_from_objc(msg_send![error, localizedRecoverySuggestion]));
|
||||||
|
println!("failure reason {}", string_from_objc(msg_send![error, localizedFailureReason]));
|
||||||
|
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
assert!(!stream.is_null());
|
||||||
|
let _: () = msg_send![stream, startCaptureWithCompletionHandler: start_capture_completion];
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let _: id = msg_send![
|
let _: id = msg_send![
|
||||||
|
@ -75,6 +120,24 @@ pub unsafe fn string_from_objc(string: id) -> String {
|
||||||
.to_string()
|
.to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern "C" fn sample_output(this: &Object, _: Sel, stream: id, buffer: id, kind: NSInteger) {
|
||||||
|
println!("sample_output");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
fn CMTimeMake(value: u64, timescale: i32) -> CMTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
struct CMTime {
|
||||||
|
value: i64,
|
||||||
|
timescale: i32,
|
||||||
|
flags: u32,
|
||||||
|
epoch: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
fn quit(_: &Quit, cx: &mut gpui::MutableAppContext) {
|
fn quit(_: &Quit, cx: &mut gpui::MutableAppContext) {
|
||||||
cx.platform().quit();
|
cx.platform().quit();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
mod atlas;
|
mod atlas;
|
||||||
mod dispatcher;
|
pub mod dispatcher;
|
||||||
mod event;
|
mod event;
|
||||||
mod fonts;
|
mod fonts;
|
||||||
mod geometry;
|
mod geometry;
|
||||||
|
|
1
styles/package-lock.json
generated
1
styles/package-lock.json
generated
|
@ -5,6 +5,7 @@
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
|
"name": "styles",
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue