Compare commits

...

6 Commits

5 changed files with 61 additions and 52 deletions

View File

@ -96,7 +96,7 @@ target_sources(nf7
$<$<PLATFORM_ID:Linux>:common/native_file_unix.cc> $<$<PLATFORM_ID:Linux>:common/native_file_unix.cc>
file/audio_context.cc file/audio_context.cc
file/audio_io.cc file/audio_device.cc
file/luajit_context.cc file/luajit_context.cc
file/luajit_node.cc file/luajit_node.cc
file/luajit_obj.cc file/luajit_obj.cc

View File

@ -42,7 +42,7 @@ struct FileCreatePopup final {
ImGui::Spacing(); ImGui::Spacing();
} }
if (ImGui::BeginListBox("type", {16*em, 4*em})) { if (ImGui::BeginListBox("type", {16*em, 8*em})) {
for (const auto& reg : nf7::File::registry()) { for (const auto& reg : nf7::File::registry()) {
const auto& t = *reg.second; const auto& t = *reg.second;

View File

@ -34,43 +34,41 @@ using namespace std::literals;
namespace nf7 { namespace nf7 {
namespace { namespace {
class IO final : public nf7::File, public nf7::DirItem, public nf7::Node { class Device final : public nf7::File, public nf7::DirItem, public nf7::Node {
public: public:
static inline const GenericTypeInfo<IO> kType = {"Audio/IO", {"DirItem",}}; static inline const GenericTypeInfo<Device> kType = {"Audio/Device", {"DirItem",}};
class Ring; class Ring;
class PlaybackLambda; class PlaybackLambda;
class CaptureLambda; class CaptureLambda;
using DeviceSelector = std::variant<size_t, std::string>; using Selector = std::variant<size_t, std::string>;
struct DeviceSelectorVisitor; struct SelectorVisitor;
static ma_device_config defaultConfig() noexcept { static ma_device_config defaultConfig() noexcept {
ma_device_config cfg; ma_device_config cfg;
cfg = ma_device_config_init(ma_device_type_playback); cfg = ma_device_config_init(ma_device_type_playback);
cfg.sampleRate = 480000; cfg.sampleRate = 48000;
cfg.playback.format = ma_format_f32; cfg.playback.format = ma_format_f32;
cfg.playback.channels = 2; cfg.playback.channels = 2;
cfg.capture.format = ma_format_f32; cfg.capture.format = ma_format_f32;
cfg.capture.channels = 2; cfg.capture.channels = 2;
return cfg; return cfg;
} }
IO(Env& env, Device(Env& env, Selector&& sel = size_t{0}, const ma_device_config& cfg = defaultConfig()) noexcept :
DeviceSelector&& sel = size_t{0},
const ma_device_config& cfg = defaultConfig()) noexcept :
File(kType, env), nf7::DirItem(DirItem::kMenu | DirItem::kTooltip), File(kType, env), nf7::DirItem(DirItem::kMenu | DirItem::kTooltip),
data_(std::make_shared<Data>()), data_(std::make_shared<Data>()),
selector_(std::move(sel)), cfg_(cfg) { selector_(std::move(sel)), cfg_(cfg) {
} }
IO(Env& env, Deserializer& ar) : IO(env) { Device(Env& env, Deserializer& ar) : Device(env) {
ar(selector_, cfg_); ar(selector_, cfg_);
} }
void Serialize(Serializer& ar) const noexcept override { void Serialize(Serializer& ar) const noexcept override {
ar(selector_, cfg_); ar(selector_, cfg_);
} }
std::unique_ptr<nf7::File> Clone(nf7::Env& env) const noexcept override { std::unique_ptr<nf7::File> Clone(nf7::Env& env) const noexcept override {
return std::make_unique<IO>(env, DeviceSelector {selector_}, cfg_); return std::make_unique<Device>(env, Selector {selector_}, cfg_);
} }
std::shared_ptr<nf7::Lambda> CreateLambda(const std::shared_ptr<nf7::Lambda::Owner>&) noexcept override; std::shared_ptr<nf7::Lambda> CreateLambda(const std::shared_ptr<nf7::Lambda::Owner>&) noexcept override;
@ -104,13 +102,13 @@ class IO final : public nf7::File, public nf7::DirItem, public nf7::Node {
std::shared_ptr<Data> data_; std::shared_ptr<Data> data_;
// persistent params // persistent params
DeviceSelector selector_; Selector selector_;
ma_device_config cfg_; ma_device_config cfg_;
// ConfigPopup param // ConfigPopup param
struct ConfigPopup final : std::enable_shared_from_this<ConfigPopup> { struct ConfigPopup final : std::enable_shared_from_this<ConfigPopup> {
ma_device_config cfg; ma_device_config cfg;
DeviceSelector selector; Selector selector;
std::atomic<bool> fetching = false; std::atomic<bool> fetching = false;
ma_device_info* devs = nullptr; ma_device_info* devs = nullptr;
@ -124,7 +122,7 @@ class IO final : public nf7::File, public nf7::DirItem, public nf7::Node {
std::make_shared<nf7::GenericContext>(f, "fetching device list"), std::make_shared<nf7::GenericContext>(f, "fetching device list"),
[this, self = shared_from_this(), mode](auto ma) { [this, self = shared_from_this(), mode](auto ma) {
try { try {
auto [ptr, n] = IO::FetchDevs(ma, mode); auto [ptr, n] = Device::FetchDevs(ma, mode);
devs = ptr; devs = ptr;
devs_n = static_cast<ma_uint32>(n); devs_n = static_cast<ma_uint32>(n);
} catch (nf7::Exception&) { } catch (nf7::Exception&) {
@ -187,7 +185,7 @@ class IO final : public nf7::File, public nf7::DirItem, public nf7::Node {
static bool UpdateModeSelector(ma_device_type*) noexcept; static bool UpdateModeSelector(ma_device_type*) noexcept;
static const ma_device_info* UpdateDeviceSelector(DeviceSelector*, ma_device_info*, size_t) noexcept; static const ma_device_info* UpdateSelector(Selector*, ma_device_info*, size_t) noexcept;
static void UpdatePresetSelector(ma_device_config*, const ma_device_info*) noexcept; static void UpdatePresetSelector(ma_device_config*, const ma_device_info*) noexcept;
@ -199,7 +197,7 @@ class IO final : public nf7::File, public nf7::DirItem, public nf7::Node {
} }
}; };
class IO::Ring final { class Device::Ring final {
public: public:
static constexpr auto kDur = 3000; /* msecs */ static constexpr auto kDur = 3000; /* msecs */
@ -282,8 +280,8 @@ class IO::Ring final {
std::atomic<uint64_t> time_ = 0; std::atomic<uint64_t> time_ = 0;
}; };
class IO::PlaybackLambda final : public nf7::Lambda, class Device::PlaybackLambda final : public nf7::Lambda,
public std::enable_shared_from_this<IO::PlaybackLambda> { public std::enable_shared_from_this<Device::PlaybackLambda> {
public: public:
static inline const std::vector<std::string> kInputs = {"get_info", "mix"}; static inline const std::vector<std::string> kInputs = {"get_info", "mix"};
static inline const std::vector<std::string> kOutputs = {"info", "mixed_size"}; static inline const std::vector<std::string> kOutputs = {"info", "mixed_size"};
@ -297,7 +295,7 @@ class IO::PlaybackLambda final : public nf7::Lambda,
}; };
PlaybackLambda() = delete; PlaybackLambda() = delete;
PlaybackLambda(IO& f, const std::shared_ptr<Owner>& owner) noexcept : PlaybackLambda(Device& f, const std::shared_ptr<Owner>& owner) noexcept :
Lambda(owner), data_(f.data_), info_(f.infoTuple()) { Lambda(owner), data_(f.data_), info_(f.infoTuple()) {
} }
@ -331,8 +329,8 @@ class IO::PlaybackLambda final : public nf7::Lambda,
uint64_t time_ = 0; uint64_t time_ = 0;
}; };
class IO::CaptureLambda final : public nf7::Lambda, class Device::CaptureLambda final : public nf7::Lambda,
std::enable_shared_from_this<IO::CaptureLambda> { std::enable_shared_from_this<Device::CaptureLambda> {
public: public:
static inline const std::vector<std::string> kInputs = {"get_info", "peek"}; static inline const std::vector<std::string> kInputs = {"get_info", "peek"};
static inline const std::vector<std::string> kOutputs = {"info", "samples"}; static inline const std::vector<std::string> kOutputs = {"info", "samples"};
@ -346,7 +344,7 @@ class IO::CaptureLambda final : public nf7::Lambda,
}; };
CaptureLambda() = delete; CaptureLambda() = delete;
CaptureLambda(IO& f, const std::shared_ptr<Owner>& owner) noexcept : CaptureLambda(Device& f, const std::shared_ptr<Owner>& owner) noexcept :
Lambda(owner), data_(f.data_), info_(f.infoTuple()) { Lambda(owner), data_(f.data_), info_(f.infoTuple()) {
} }
@ -378,28 +376,28 @@ class IO::CaptureLambda final : public nf7::Lambda,
std::optional<uint64_t> time_; std::optional<uint64_t> time_;
}; };
std::shared_ptr<nf7::Lambda> IO::CreateLambda( std::shared_ptr<nf7::Lambda> Device::CreateLambda(
const std::shared_ptr<nf7::Lambda::Owner>& owner) noexcept { const std::shared_ptr<nf7::Lambda::Owner>& owner) noexcept {
switch (cfg_.deviceType) { switch (cfg_.deviceType) {
case ma_device_type_playback: case ma_device_type_playback:
return std::make_shared<IO::PlaybackLambda>(*this, owner); return std::make_shared<Device::PlaybackLambda>(*this, owner);
case ma_device_type_capture: case ma_device_type_capture:
return std::make_shared<IO::CaptureLambda>(*this, owner); return std::make_shared<Device::CaptureLambda>(*this, owner);
default: default:
std::abort(); std::abort();
} }
} }
struct IO::DeviceSelectorVisitor final { struct Device::SelectorVisitor final {
public: public:
DeviceSelectorVisitor() = delete; SelectorVisitor() = delete;
DeviceSelectorVisitor(ma_device_info* info, size_t n) noexcept : info_(info), n_(n) { SelectorVisitor(ma_device_info* info, size_t n) noexcept : info_(info), n_(n) {
} }
DeviceSelectorVisitor(const DeviceSelectorVisitor&) = delete; SelectorVisitor(const SelectorVisitor&) = delete;
DeviceSelectorVisitor(DeviceSelectorVisitor&&) = delete; SelectorVisitor(SelectorVisitor&&) = delete;
DeviceSelectorVisitor& operator=(const DeviceSelectorVisitor&) = delete; SelectorVisitor& operator=(const SelectorVisitor&) = delete;
DeviceSelectorVisitor& operator=(DeviceSelectorVisitor&&) = delete; SelectorVisitor& operator=(SelectorVisitor&&) = delete;
ma_device_info* operator()(const size_t& idx) noexcept { ma_device_info* operator()(const size_t& idx) noexcept {
return idx < n_? &info_[idx]: nullptr; return idx < n_? &info_[idx]: nullptr;
@ -418,7 +416,7 @@ struct IO::DeviceSelectorVisitor final {
}; };
void IO::InitDev() noexcept { void Device::InitDev() noexcept {
if (!data_->aq) { if (!data_->aq) {
data_->log.Error("audio queue is missing"); data_->log.Error("audio queue is missing");
return; return;
@ -446,7 +444,7 @@ void IO::InitDev() noexcept {
} }
auto [devs, devs_n] = FetchDevs(ma, cfg.deviceType); auto [devs, devs_n] = FetchDevs(ma, cfg.deviceType);
auto dinfo = std::visit(DeviceSelectorVisitor {devs, devs_n}, sel); auto dinfo = std::visit(SelectorVisitor {devs, devs_n}, sel);
if (!dinfo) { if (!dinfo) {
throw nf7::Exception("missing device"); throw nf7::Exception("missing device");
} }
@ -475,7 +473,7 @@ void IO::InitDev() noexcept {
--d->busy; --d->busy;
}); });
} }
void IO::DeinitDev() noexcept { void Device::DeinitDev() noexcept {
if (!data_->aq) { if (!data_->aq) {
data_->log.Error("audio queue is missing"); data_->log.Error("audio queue is missing");
return; return;
@ -491,7 +489,7 @@ void IO::DeinitDev() noexcept {
--d->busy; --d->busy;
}); });
} }
void IO::BuildNode() noexcept { void Device::BuildNode() noexcept {
switch (cfg_.deviceType) { switch (cfg_.deviceType) {
case ma_device_type_playback: case ma_device_type_playback:
nf7::Node::input_ = PlaybackLambda::kInputs; nf7::Node::input_ = PlaybackLambda::kInputs;
@ -508,7 +506,7 @@ void IO::BuildNode() noexcept {
} }
void IO::Handle(const Event& ev) noexcept { void Device::Handle(const Event& ev) noexcept {
switch (ev.type) { switch (ev.type) {
case Event::kAdd: case Event::kAdd:
data_->log.SetUp(*this); data_->log.SetUp(*this);
@ -537,7 +535,7 @@ void IO::Handle(const Event& ev) noexcept {
} }
void IO::Update() noexcept { void Device::Update() noexcept {
if (const auto popup = std::exchange(popup_, nullptr)) { if (const auto popup = std::exchange(popup_, nullptr)) {
ImGui::OpenPopup(popup); ImGui::OpenPopup(popup);
} }
@ -565,7 +563,7 @@ void IO::Update() noexcept {
} }
const ma_device_info* dev = nullptr; const ma_device_info* dev = nullptr;
if (!p->fetching) { if (!p->fetching) {
dev = UpdateDeviceSelector(&p->selector, p->devs, p->devs_n); dev = UpdateSelector(&p->selector, p->devs, p->devs_n);
} else { } else {
ImGui::LabelText("device", "fetching list..."); ImGui::LabelText("device", "fetching list...");
} }
@ -588,7 +586,7 @@ void IO::Update() noexcept {
ImGui::EndPopup(); ImGui::EndPopup();
} }
} }
void IO::UpdateMenu() noexcept { void Device::UpdateMenu() noexcept {
if (cfg_.deviceType == ma_device_type_playback) { if (cfg_.deviceType == ma_device_type_playback) {
if (ImGui::MenuItem("simulate sinwave output for 1 sec")) { if (ImGui::MenuItem("simulate sinwave output for 1 sec")) {
const auto wave = GenerateSineWave(cfg_.sampleRate, cfg_.playback.channels); const auto wave = GenerateSineWave(cfg_.sampleRate, cfg_.playback.channels);
@ -600,7 +598,7 @@ void IO::UpdateMenu() noexcept {
popup_ = "ConfigPopup"; popup_ = "ConfigPopup";
} }
} }
void IO::UpdateTooltip() noexcept { void Device::UpdateTooltip() noexcept {
const char* mode = const char* mode =
cfg_.deviceType == ma_device_type_playback? "playback": cfg_.deviceType == ma_device_type_playback? "playback":
cfg_.deviceType == ma_device_type_capture ? "capture": cfg_.deviceType == ma_device_type_capture ? "capture":
@ -611,7 +609,7 @@ void IO::UpdateTooltip() noexcept {
ImGui::Text("channels : %" PRIu32, cfg_.playback.channels); ImGui::Text("channels : %" PRIu32, cfg_.playback.channels);
ImGui::Text("sample rate: %" PRIu32, cfg_.sampleRate); ImGui::Text("sample rate: %" PRIu32, cfg_.sampleRate);
} }
bool IO::UpdateModeSelector(ma_device_type* m) noexcept { bool Device::UpdateModeSelector(ma_device_type* m) noexcept {
const char* mode = const char* mode =
*m == ma_device_type_playback? "playback": *m == ma_device_type_playback? "playback":
*m == ma_device_type_capture? "capture": *m == ma_device_type_capture? "capture":
@ -630,9 +628,9 @@ bool IO::UpdateModeSelector(ma_device_type* m) noexcept {
} }
return ret; return ret;
} }
const ma_device_info* IO::UpdateDeviceSelector( const ma_device_info* Device::UpdateSelector(
DeviceSelector* sel, ma_device_info* devs, size_t n) noexcept { Selector* sel, ma_device_info* devs, size_t n) noexcept {
const auto dev = std::visit(DeviceSelectorVisitor {devs, n}, *sel); const auto dev = std::visit(SelectorVisitor {devs, n}, *sel);
if (ImGui::BeginCombo("device", dev? dev->name: "(missing)")) { if (ImGui::BeginCombo("device", dev? dev->name: "(missing)")) {
for (size_t i = 0; i < n; ++i) { for (size_t i = 0; i < n; ++i) {
@ -665,7 +663,7 @@ const ma_device_info* IO::UpdateDeviceSelector(
} }
return dev; return dev;
} }
void IO::UpdatePresetSelector(ma_device_config* cfg, const ma_device_info* dev) noexcept { void Device::UpdatePresetSelector(ma_device_config* cfg, const ma_device_info* dev) noexcept {
auto& srate = cfg->sampleRate; auto& srate = cfg->sampleRate;
auto& ch = GetChannels(*cfg); auto& ch = GetChannels(*cfg);

View File

@ -45,7 +45,7 @@ class Imm final : public nf7::File, public nf7::DirItem, public nf7::Node {
{kStringText, "string/text"}, {kStringText, "string/text"},
}; };
Imm(Env& env, Type type = kPulse, nf7::Value&& v = {}) noexcept : Imm(Env& env, Type type = kInteger, nf7::Value&& v = nf7::Value::Integer {0}) noexcept :
File(kType, env), DirItem(DirItem::kNone), mem_(*this, {type, std::move(v)}) { File(kType, env), DirItem(DirItem::kNone), mem_(*this, {type, std::move(v)}) {
output_ = {"out"}; output_ = {"out"};
} }

View File

@ -135,6 +135,7 @@ class Network final : public nf7::File,
std::vector<std::weak_ptr<Network::Lambda>> lambdas_running_; std::vector<std::weak_ptr<Network::Lambda>> lambdas_running_;
const char* popup_ = nullptr; const char* popup_ = nullptr;
ImVec2 canvas_action_pos_;
// persistent params // persistent params
gui::Window win_; gui::Window win_;
@ -1097,7 +1098,10 @@ void Network::Update() noexcept {
if (p.Update(*this)) { if (p.Update(*this)) {
auto item = std::make_unique<Item>(next_++, p.type().Create(env())); auto item = std::make_unique<Item>(next_++, p.type().Create(env()));
auto ctx = std::make_shared<nf7::GenericContext>(*this, "adding new node"); auto ctx = std::make_shared<nf7::GenericContext>(*this, "adding new node");
auto& item_ref = *item;
QueueCommand(ctx, std::make_unique<Item::SwapCommand>(*this, std::move(item))); QueueCommand(ctx, std::make_unique<Item::SwapCommand>(*this, std::move(item)));
QueueCommand(ctx, std::make_unique<Item::MoveCommand>(item_ref, canvas_action_pos_));
} }
ImGui::EndPopup(); ImGui::EndPopup();
} }
@ -1359,7 +1363,8 @@ void Network::Update() noexcept {
ImGui::EndGroup(); ImGui::EndGroup();
// ---- editor window / canvas // ---- editor window / canvas
if (ImGui::BeginChild("canvas")) { if (ImGui::BeginChild("canvas", {0, 0}, false, ImGuiWindowFlags_NoMove)) {
const auto canvas_pos = ImGui::GetCursorScreenPos();
ImNodes::BeginCanvas(&canvas_); ImNodes::BeginCanvas(&canvas_);
// update child nodes // update child nodes
@ -1398,7 +1403,6 @@ void Network::Update() noexcept {
auto cmd = NodeLinkStore::SwapCommand::CreateToAdd(links_, std::move(lk)); auto cmd = NodeLinkStore::SwapCommand::CreateToAdd(links_, std::move(lk));
QueueCommand(ctx, std::move(cmd)); QueueCommand(ctx, std::move(cmd));
} }
ImNodes::EndCanvas(); ImNodes::EndCanvas();
// popup menu for canvas // popup menu for canvas
@ -1406,9 +1410,16 @@ void Network::Update() noexcept {
ImGuiPopupFlags_MouseButtonRight | ImGuiPopupFlags_MouseButtonRight |
ImGuiPopupFlags_NoOpenOverExistingPopup; ImGuiPopupFlags_NoOpenOverExistingPopup;
if (ImGui::BeginPopupContextWindow(nullptr, kFlags)) { if (ImGui::BeginPopupContextWindow(nullptr, kFlags)) {
if (ImGui::IsWindowAppearing()) {
canvas_action_pos_ =
(ImGui::GetMousePos() - canvas_pos - canvas_.Offset)/canvas_.Zoom;
}
if (ImGui::MenuItem("add")) { if (ImGui::MenuItem("add")) {
popup_ = "AddPopup"; popup_ = "AddPopup";
} }
if (ImGui::MenuItem("I/O sockets")) {
popup_ = "SocketEditorPopup";
}
ImGui::Separator(); ImGui::Separator();
if (ImGui::MenuItem("undo", nullptr, false, !!history_.prev())) { if (ImGui::MenuItem("undo", nullptr, false, !!history_.prev())) {
UnDo(); UnDo();
@ -1417,8 +1428,8 @@ void Network::Update() noexcept {
ReDo(); ReDo();
} }
ImGui::Separator(); ImGui::Separator();
if (ImGui::MenuItem("I/O sockets")) { if (ImGui::MenuItem("reset canvas zoom")) {
popup_ = "SocketEditorPopup"; canvas_.Zoom = 1.f;
} }
ImGui::EndPopup(); ImGui::EndPopup();
} }