Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ctrlmidich option for "own channel" regardless of number #3394

Conversation

pljones
Copy link
Collaborator

@pljones pljones commented Oct 2, 2024

Short description of changes

From @AndersGoran.

This adds a ;z option to --ctrlmidich parsing that adds control of the client channel, regardless of its channel number. This applies to all the channel-specific controls used. It takes up one extra MIDI controller number for each.

CHANGELOG: Add ctrlmidich option for "own channel" regardless of number

Context: Fixes an issue?

Raised in https://github.com/orgs/jamulussoftware/discussions/2220#discussioncomment-10811813

Does this change need documentation? What needs to be documented and how?

Yes, but it needs reviewing to see if this is how we want to do it first.

Status of this Pull Request

Proof of concept.

What is missing until this pull request can be merged?

Needs reviewing and considering whether "own client id" concept should apply to all controls (e.g. use ;z as a switch to bump lock the first CC number for each control to "own client id").

Checklist

  • I've verified that this Pull Request follows the general code principles
  • I tested my code and it does what I want
  • My code follows the style guide
  • I waited some time after this Pull Request was opened and all GitHub checks completed without errors.
  • I've filled all the content above

AUTOBUILD: Please build all targets

@pljones
Copy link
Collaborator Author

pljones commented Oct 2, 2024

There may currently be a problem with CSoundBase::ParseCommandLineArgument in that it doesn't necessarily set every element of aMidiCtls for each sMidiCtlChar but it doesn't check in CSoundBase::ParseMIDIMessage that the element exists.

So I'd like the ParseCommandLineArgument changed to ensure each possible option gets set to some "unset" default value and then overridden by the command line argument.

Separately, I'd like ;z<n> changed to just ;z and have that toggle "Use Own Channel Id In First CC For Control" mode -- so it's not just Fader that gets the benefit, it's Pan and Mute, too. Or have ;z<n> with <n> being the offset into the usual range of CC numbers for each control.

@pljones pljones requested review from softins and ann0see October 2, 2024 18:09
@pljones pljones self-assigned this Oct 2, 2024
@pljones pljones added feature request Feature request good first issue Things which should be doable without lots of context labels Oct 2, 2024
@pljones pljones added this to the Release 3.12.0 milestone Oct 2, 2024
@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch from 2ade0ca to a8bb22e Compare October 2, 2024 18:13
@ann0see ann0see requested a review from ignotus666 October 2, 2024 18:36
src/sound/soundbase.h Outdated Show resolved Hide resolved
src/audiomixerboard.cpp Outdated Show resolved Hide resolved
@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch 2 times, most recently from 18c8889 to c95a810 Compare October 4, 2024 19:26
@pljones
Copy link
Collaborator Author

pljones commented Oct 4, 2024

OK, so I decided to make the change I suggested.

Once the build is complete for Windows, I'll give it a try here.


The legacy definition has just one or two numbers that only provides a definition for the controller offset of the fader controllers (default 70 for the sake of Behringer X-Touch)

[MIDI channel];[offset for first fader]

The more verbose new form is a sequence of offsets and counts for various controllers:

[MIDI channel];[control letter][offset]*[count](;...)

Currently, the following control letters are defined:

Control control letter
Fader f
Pan p
Solo s
Mute m
  • offset is the base MIDI CC number for the control.
  • count is the number of CC values for the control (i.e. the number Jamulus channels that can be controlled).

Additionally, o has a single offset (i.e. count is ignored and taken as 1) and controls Mute Myself.

In addition, z reserves the first CC number for a control to mean "my Jamulus channel". For example

--ctrlmidich '1;f0*9;z'

would mean MIDI CC0 controlled my Jamulus channel fader, with CC1 to CC8 for Jamulus channels 1 to 8 (so if you were channel 6, you get two MIDI CC controls).

An example for a Korg nanoKONTROL2 with 8 fader controllers starting at offset 0 and 8 pan controllers starting at offset 16
would be

[MIDI channel];f0*8;p16*8

However, at the current point of time only 'f' and 'p' controllers are actually implemented. // may not be true...

@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch 2 times, most recently from 76af113 to d790eb1 Compare October 4, 2024 19:37
src/sound/soundbase.h Outdated Show resolved Hide resolved
src/sound/soundbase.cpp Outdated Show resolved Hide resolved
@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch from d790eb1 to 109d4eb Compare October 4, 2024 19:52
@pljones
Copy link
Collaborator Author

pljones commented Oct 5, 2024

Ugh, failure due to qt5 -> qt6. Investigating.


Fixed (more a compiler issue, I think).

@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch 4 times, most recently from 42a80ac to 713cfbc Compare October 5, 2024 09:04
@pljones
Copy link
Collaborator Author

pljones commented Oct 5, 2024

OK, I can't get JACK to run without crashing on Windows any more for some reason. So I can't actually test this (as Jamulus ASIO doesn't support MIDI, as far as I know, and I've no MIDI - or audio - on my Linux box).

@pljones pljones added the needs documentation PRs requiring documentation changes or additions label Oct 5, 2024
@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch from 713cfbc to 3f0983e Compare October 5, 2024 16:32
@pljones
Copy link
Collaborator Author

pljones commented Oct 5, 2024

@AndersGoran could you check whether the latest build is useable?

It should work like this:

--ctrlmidich 0;z;f0*9 gives you MIDI CC 0 controlling "my" fader, regardless of server-assigned Jamulus channel number, MIDI CC 1 controlling Jamulus channel 0 through to MIDI CC 8 controlling Jamulus channel 7.

It should work for all the Jamulus controls.

--ctrlmidich 0;z;f0;p1;s2;m3;o4 should give you control only over your own channel for fader, pan, mute (um, no, don't) and mute myself (better). --ctrlmidich 0;z;f0*9;p1*9;s2*9;m3*9;o4 gives you your own channel, plus additional controls for up to eight Jamulus channels, which could include your own channel. (If you were on Jamulus channel 3, you'd get MIDI CC 0 and MIDI CC 4 controlling your fader, for example; if you were on Jamulus channel 17, you'd still get control over your own channel and the first eight Jamulus channels.)

Note: remember, having control over your own channel only affects what you hear, not what others hear.

@AndersGoran
Copy link

I cloned https://github.com/pljones/jamulus.git, current HEAD is 3f0983e, compiles fine (I'm on macOS) but I'm afraid the "z" flag makes no difference. As before, my controller sends CC 100-103 from four separate pedals, so I use "0;z;f100*4" and the 100 pedal controls channel 0, 101 control 1, and so on. When I'm on e.g. 1, I have to use the pedal that sends 101 to move my fader.

Is there a pre-built executable somewhere that I should be using?

Copy link
Member

@ann0see ann0see left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have the feeling that documentation is lacking on explaining the code.
I feel unsure about the exact functionality - especially the cli parsing.

@pljones
Copy link
Collaborator Author

pljones commented Oct 15, 2024

I have the feeling that documentation is lacking on explaining the code. I feel unsure about the exact functionality - especially the cli parsing.

Which documentation?

What are you unsure about?

@ann0see
Copy link
Member

ann0see commented Oct 17, 2024

I think I need to go over the code in my editor. The diff on GitHub doesn't show me enough.

@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch 2 times, most recently from cd31d25 to e1aab44 Compare October 29, 2024 18:19
@softins
Copy link
Member

softins commented Oct 30, 2024

I know I'm rather late to the party here, due to limited time, but have today been studying and thinking about this solution to @AndersGoran's original issue.

I don't like the fact that when using the z option, there will be two faders that control the user's own channel: the first fader (by virtue of the z option), and the fader assigned to the user's actual channel. This seems user-unfriendly, wasteful of a physical fader, and with the potential for confusion, if the two faders, pans or mute/solo buttons are set to different states.

I would like to propose a different solution (apologies for not doing so until now!):

  • The user's own channel always gets given the internal fader channel 0, whatever client ID it is on. So my fader would always look like 0:TonyM.
  • Any other channels with a client ID less than my own client ID will be given the internal fader channel of ID+1.
  • Any other channels with a client ID greater than my own client ID will be given the internal fader channel equal to their ID as at present.

So a session where I am TonyM might look like this:

Client ID on server Name Fader ID Tag on mixer
0 Tom 1 1:Tom
1 Dick 2 2:Dick
2 TonyM 0 0:TonyM
3 Harry 3 3:Harry
4 Jim 4 4:Jim

That way, there is only one fader channel per client, and my own fader is always the first one. These fader IDs are local to my own Jamulus client, and don't need to be the same as those on any other user's client.

Since the user never has any control over which client IDs are assigned by the server, it doesn't matter that the other user's fader number doesn't necessarily equal their client ID, and the above technique would ensure the first MIDI fader is always the user's own, since they always get fader channel 0. This wouldn't even need the extra z flag in the --ctrlmidich string; there would be no downside in making the above always the case.

This would also work well with @pljones recent "Sort by Channel" option in #3418 (where Channel is the fader ID).

Comments?

@pljones
Copy link
Collaborator Author

pljones commented Oct 30, 2024

It's probably best if @AndersGoran comments.

@ann0see
Copy link
Member

ann0see commented Oct 30, 2024

@softins I agree.

The user's own channel always gets given the internal fader channel 0, whatever client ID it is on. So my fader would always look like 0:TonyM.

This could potentially also simply the own fader first implementation

@AndersGoran
Copy link

Thank you @softins, for looking and thinking about this. Your suggestion sounds good to me.

My use case has always been such that I want a predictable way to assign one single MIDI CC to always control my fader. I don't really care about any other faders than my own, in any of the groups I play in on Jamulus.

I just want to be able to mute myself in my own mix with MIDI CC. If I could trust that the first CC given in --ctrlmidich always controls my fader, that's all I need.

@pljones
Copy link
Collaborator Author

pljones commented Oct 31, 2024

So if I've got five sliders assigned and I always want slide 1 to be "me", I'd expect 2, 3, 4 and 5 to be someone else, agreed.

I'd also expect them to be in order on the mixer board as my hardware controller, with "me" on the left -- which is what "Sort by Server Channel" would do, normally, if you "overwrote" the Server channel with the renumbered value like this.

What I wouldn't want is for the mixer and hardware order to mismatch - 2 3 0 4 5 vs 1 2 3 4 5

(And, of course, if channel 4 disconnects, controller 4 wouldn't control anything - I'd have to see "5" and know what it meant.)

Indeed, "Own channel first" could always do this -- after numbering the channels according to the sort, renumber with the proposed rule, before displaying. (Currently it moves the channel from where it is to the first position, regardless - same effect, different method.)

@softins
Copy link
Member

softins commented Oct 31, 2024

I'm in the middle of doing a proof of concept for my suggestion, which we will be able to try out in a few days. It's quite complicated working out the best way, due to the onion-like many layers in the software structure: Client; ClientDlg; MainMixerBoard; Faders; with many signal/slot connections between them.

One thing we can't do is overwrite the iChanID, since that by definition must match the channel ID on the server. So the MIDI channel numbers must be stored or calculated independently on the client side, and the mapping between them and channel IDs done either algorithmically or via lookup tables. I'm currently trying the former approach as per my suggestion further up.

@ann0see
Copy link
Member

ann0see commented Nov 1, 2024

Lookup table sounds more performant and less messy.

@pljones
Copy link
Collaborator Author

pljones commented Nov 1, 2024

Lookup table sounds more performant and less messy.

Yup - the current version uses a lookup table.

However you do it, you need to cater for both the situation where the user wants the reserved channel and when they don't want it. (It's perfectly reasonable for someone to want controller three, or whatever, to control channel 3, regardless of whether it's their own channel or someone else's.)

But at the time you parse the controller values, you can't populate the map from controller to channel for "my channel", as you don't know what number it will be, so if you don't want to reserve a slot for it, you need another map on top of the existing one, I think, remapping the numbers every time someone joins (or leaves, though that's not important). There's currently something happening in the UI control to keep the mixer channels in order -- but, really, I'd rather MIDI control of the Client was not dependent upon the UI being enabled.

It's getting to the point I'm thinking "We've got CClientDlg and CClientRpc both controlling CClient, why not get CClientMidi, too, for MIDI control, and give it MIDI output, too, to reflect changes?" Overhaul CClientDlg, CClientRpc and CClient to communicate properly over the signal/slot interfaces (only, as far as I'm concerned -- CClient would define public slots for controls and emit signals with changes in state - that it held -- and would have public read-only calls to read state).

@pljones pljones force-pushed the AndersGoran/ctrlmidich-for-own-channel branch from e1aab44 to 847fc24 Compare November 2, 2024 13:24
@pljones
Copy link
Collaborator Author

pljones commented Nov 3, 2024

OK, got to test this on Windows (develop branch build of jackd needed to fix MIDI on Windows). The current build is doing what I'd intended.

@ann0see ann0see self-requested a review November 3, 2024 21:41
@pljones pljones marked this pull request as draft November 24, 2024 14:40
@pljones
Copy link
Collaborator Author

pljones commented Nov 24, 2024

Set to draft as this will get reworked / dropped after #3426 merges.

@pljones
Copy link
Collaborator Author

pljones commented Dec 4, 2024

OK, closing this in favour of #3426

@pljones pljones closed this Dec 4, 2024
@pljones pljones removed this from the Release 3.12.0 milestone Dec 4, 2024
@pljones pljones deleted the AndersGoran/ctrlmidich-for-own-channel branch December 10, 2024 18:32
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
feature request Feature request good first issue Things which should be doable without lots of context needs documentation PRs requiring documentation changes or additions
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants