Skip to content

Commit 1511835

Browse files
authored
Kotlinify ControlWrapper and Shells (#27)
2 parents 95e1d5e + 8412103 commit 1511835

File tree

6 files changed

+594
-614
lines changed

6 files changed

+594
-614
lines changed

CHANGES.md

+2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
# DurianSwt releases
22

33
## [Unreleased]
4+
### Changed
5+
- Convert `ControlWrapper` and `Shells` to kotlin for better nullability. ([#27](https://github.com/diffplug/durian-swt/pull/27))
46

57
## [4.3.0] - 2023-12-08
68
### Added

durian-swt/src/main/java/com/diffplug/common/swt/ControlWrapper.java

-145
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,126 @@
1+
/*
2+
* Copyright 2020 DiffPlug
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package com.diffplug.common.swt
17+
18+
import org.eclipse.swt.widgets.Composite
19+
import org.eclipse.swt.widgets.Control
20+
import org.eclipse.swt.widgets.Shell
21+
22+
23+
/**
24+
* Wraps an SWT [Control] to encapsulate its API.
25+
*
26+
*
27+
* The traditional way to make a custom class is this: `class CustomControl extends [Composite]`
28+
*
29+
*
30+
* This has three main problems:
31+
*
32+
* 1. Users can add random widgets to your "Control" because it exposes the [Composite] interface.
33+
* 1. Users can set the layout to your "Control" because it exposes the [Composite] interface.
34+
* 1. Users can add random listeners to your "Control", and overridding [Widget.addListener][org.eclipse.swt.widgets.Widget.addListener] to intercept them is a **very dangerous plan**.
35+
*
36+
*
37+
*
38+
* ControlWrapper fixes this by providing an low-overhead skeleton which encapsulates the
39+
* SWT Control that you're using as the base of your custom control, which allows you to only
40+
* expose the APIs that are appropriate.
41+
*/
42+
interface ControlWrapper {
43+
var layoutData: Any?
44+
/** Returns the LayoutData for this control. */
45+
get() = rootControl.layoutData
46+
/** Sets the LayoutData for this control. */
47+
set(layoutData) {
48+
rootControl.layoutData = layoutData
49+
}
50+
51+
val parent: Composite
52+
/** Returns the parent of this Control. */
53+
get() = rootControl.parent
54+
55+
val shell: Shell
56+
/** Returns the parent Shell of this Control. */
57+
get() = rootControl.shell
58+
59+
/** Disposes the underlying control. */
60+
fun dispose() {
61+
rootControl.dispose()
62+
}
63+
64+
val isDisposed: Boolean
65+
/** Returns true iff the underlying control is disposed. */
66+
get() = rootControl.isDisposed
67+
68+
/**
69+
* Changes the parent of the widget to be the one provided.
70+
* Returns `true` if the parent is successfully changed
71+
*/
72+
fun setParent(parent: Composite): Boolean {
73+
return rootControl.setParent(parent)
74+
}
75+
76+
/**
77+
* Returns the wrapped [Control] (only appropriate for limited purposes!).
78+
*
79+
*
80+
* The implementor of this ControlWrapper is free to change the wrapped Control
81+
* as she sees fit, and she doesn't have to tell you about it! You shouldn't rely
82+
* on this control being anything in particular.
83+
*
84+
*
85+
* You *can* rely on this Control for:
86+
*
87+
* 1. Managing lifetimes: `wrapped.getRootControl().addListener(SWT.Dispose, ...`
88+
*
89+
*
90+
*
91+
* But that's all. If you use it for something else, it's on you when it breaks.
92+
*/
93+
val rootControl: Control
94+
95+
/** Default implementation of a [ControlWrapper] which wraps a [Control]. */
96+
open class AroundControl<T : Control>
97+
/** Creates a ControlWrapper which wraps the given control. */(
98+
/** The wrapped control. */
99+
@JvmField protected val wrapped: T
100+
) : ControlWrapper {
101+
override val rootControl: T
102+
get() = wrapped
103+
}
104+
105+
/** Default implementation of a [ControlWrapper] which wraps some other form of `ControlWrapper` with a new interface. */
106+
open class AroundWrapper<T : ControlWrapper>(
107+
@JvmField
108+
protected val wrapped: T
109+
) : ControlWrapper {
110+
override val rootControl: Control
111+
get() = wrapped.rootControl
112+
}
113+
114+
/** Creates a ControlWrapper which wraps the given control. */
115+
class Transparent<T : Control>(
116+
override val rootControl: T
117+
) : ControlWrapper
118+
119+
companion object {
120+
/** Most-efficient way to transparently pass a Control to a ControlWrapper API. */
121+
@JvmStatic
122+
fun <T : Control> transparent(control: T): Transparent<T> {
123+
return Transparent(control)
124+
}
125+
}
126+
}

0 commit comments

Comments
 (0)