outputconfigurationstore: don't overwrite output settings with kscreen configs

The existing code overwrites all per-output data with the config from KScreen
every time an output config is found that isn't in the KWin data base yet.
That causes problems like the scale being reset to 1.0 if the user logged
into Xorg before, or resetting it to whatever scale KScreen had saved on
Wayland, which isn't much better.

Instead of doing that, use the KScreen config as inspiration for some specific
output settings, instead of copying it outright. It's not ideal, but it's
better than overwriting settings the user has explicitly set up

BUG: 485353
This commit is contained in:
Xaver Hugl 2024-04-22 22:25:29 +02:00
parent 38694bbe1a
commit 7949032411

View file

@ -52,12 +52,6 @@ std::optional<std::tuple<OutputConfiguration, QList<Output *>, OutputConfigurati
storeConfig(relevantOutputs, isLidClosed, config, order); storeConfig(relevantOutputs, isLidClosed, config, order);
return std::make_tuple(config, order, ConfigType::Preexisting); return std::make_tuple(config, order, ConfigType::Preexisting);
} }
if (auto kscreenConfig = KScreenIntegration::readOutputConfig(relevantOutputs, KScreenIntegration::connectedOutputsHash(relevantOutputs, isLidClosed))) {
auto &[config, order] = *kscreenConfig;
applyOrientationReading(config, relevantOutputs, orientation, isTabletMode);
storeConfig(relevantOutputs, isLidClosed, config, order);
return std::make_tuple(config, order, ConfigType::Preexisting);
}
auto [config, order] = generateConfig(relevantOutputs, isLidClosed); auto [config, order] = generateConfig(relevantOutputs, isLidClosed);
applyOrientationReading(config, relevantOutputs, orientation, isTabletMode); applyOrientationReading(config, relevantOutputs, orientation, isTabletMode);
storeConfig(relevantOutputs, isLidClosed, config, order); storeConfig(relevantOutputs, isLidClosed, config, order);
@ -397,12 +391,16 @@ std::pair<OutputConfiguration, QList<Output *>> OutputConfigurationStore::genera
return *closedConfig; return *closedConfig;
} }
} }
const auto kscreenConfig = KScreenIntegration::readOutputConfig(outputs, KScreenIntegration::connectedOutputsHash(outputs, isLidClosed));
OutputConfiguration ret; OutputConfiguration ret;
QList<Output *> outputOrder; QList<Output *> outputOrder;
QPoint pos(0, 0); QPoint pos(0, 0);
for (const auto output : outputs) { for (const auto output : outputs) {
const auto kscreenChangeSetPtr = kscreenConfig ? kscreenConfig->first.constChangeSet(output) : nullptr;
const auto kscreenChangeSet = kscreenChangeSetPtr ? *kscreenChangeSetPtr : OutputChangeSet{};
const auto outputIndex = findOutput(output, outputs); const auto outputIndex = findOutput(output, outputs);
const bool enable = !isLidClosed || !output->isInternal() || outputs.size() == 1; const bool enable = kscreenChangeSet.enabled.value_or(!isLidClosed || !output->isInternal() || outputs.size() == 1);
const OutputState existingData = outputIndex ? m_outputs[*outputIndex] : OutputState{}; const OutputState existingData = outputIndex ? m_outputs[*outputIndex] : OutputState{};
const auto modes = output->modes(); const auto modes = output->modes();
@ -411,19 +409,21 @@ std::pair<OutputConfiguration, QList<Output *>> OutputConfigurationStore::genera
&& mode->size() == existingData.mode->size && mode->size() == existingData.mode->size
&& mode->refreshRate() == existingData.mode->refreshRate; && mode->refreshRate() == existingData.mode->refreshRate;
}); });
const auto mode = modeIt == modes.end() ? output->currentMode() : *modeIt; const auto mode = modeIt == modes.end() ? kscreenChangeSet.mode.value_or(output->currentMode()).lock() : *modeIt;
const auto changeset = ret.changeSet(output); const auto changeset = ret.changeSet(output);
*changeset = { *changeset = {
.mode = mode, .mode = mode,
.enabled = enable, .enabled = kscreenChangeSet.enabled.value_or(enable),
.pos = pos, .pos = pos,
// kscreen scale is unreliable because it gets overwritten with the value 1 on Xorg,
// and we don't know if it's from Xorg or the 5.27 Wayland session... so just ignore it
.scale = existingData.scale.value_or(chooseScale(output, mode.get())), .scale = existingData.scale.value_or(chooseScale(output, mode.get())),
.transform = existingData.transform.value_or(output->panelOrientation()), .transform = existingData.transform.value_or(kscreenChangeSet.transform.value_or(output->panelOrientation())),
.manualTransform = existingData.manualTransform.value_or(output->panelOrientation()), .manualTransform = existingData.manualTransform.value_or(kscreenChangeSet.transform.value_or(output->panelOrientation())),
.overscan = existingData.overscan.value_or(0), .overscan = existingData.overscan.value_or(kscreenChangeSet.overscan.value_or(0)),
.rgbRange = existingData.rgbRange.value_or(Output::RgbRange::Automatic), .rgbRange = existingData.rgbRange.value_or(kscreenChangeSet.rgbRange.value_or(Output::RgbRange::Automatic)),
.vrrPolicy = existingData.vrrPolicy.value_or(VrrPolicy::Automatic), .vrrPolicy = existingData.vrrPolicy.value_or(kscreenChangeSet.vrrPolicy.value_or(VrrPolicy::Automatic)),
.highDynamicRange = existingData.highDynamicRange.value_or(false), .highDynamicRange = existingData.highDynamicRange.value_or(false),
.sdrBrightness = existingData.sdrBrightness.value_or(200), .sdrBrightness = existingData.sdrBrightness.value_or(200),
.wideColorGamut = existingData.wideColorGamut.value_or(false), .wideColorGamut = existingData.wideColorGamut.value_or(false),
@ -435,7 +435,16 @@ std::pair<OutputConfiguration, QList<Output *>> OutputConfigurationStore::genera
outputOrder.push_back(output); outputOrder.push_back(output);
} }
} }
return std::make_pair(ret, outputs); if (kscreenConfig && kscreenConfig->second.size() == outputOrder.size()) {
// make sure the old output order is consistent with the enablement states of the outputs
const bool consistent = std::ranges::all_of(outputOrder, [&kscreenConfig](const auto output) {
return kscreenConfig->second.contains(output);
});
if (consistent) {
outputOrder = kscreenConfig->second;
}
}
return std::make_pair(ret, outputOrder);
} }
std::shared_ptr<OutputMode> OutputConfigurationStore::chooseMode(Output *output) const std::shared_ptr<OutputMode> OutputConfigurationStore::chooseMode(Output *output) const