Skip to content

Commit d978b37

Browse files
committed
Improve CONTRIBUTING.md
1 parent 5b591d2 commit d978b37

File tree

1 file changed

+80
-39
lines changed

1 file changed

+80
-39
lines changed

CONTRIBUTING.md

+80-39
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,13 @@
1-
# Contributing guidelines
1+
# Contribution Guidelines
22

33
## Introduction
44

55
`egui` has been an on-and-off weekend project of mine since late 2018. I am grateful to any help I can get, but bare in mind that sometimes I can be slow to respond because I am busy with other things!
66

77
/ Emil
88

9+
## How to contribute to egui
10+
You want to contribute to egui, but don't know how? First of all: thank you! I created a special issue just for that: <https://github.com/emilk/egui/issues/3742>. But make sure you still read this file first :)
911

1012
## Discussion
1113

@@ -25,7 +27,7 @@ If you are filing a bug, please provide a way to reproduce it.
2527

2628
## Making a PR
2729

28-
First file an issue (or find an existing one) and announce that you plan to work on something. That way we will avoid having several people doing double work. Please ask for feedback before you start working on something non-trivial!
30+
For small things, just go ahead an open a PR. For bigger things, please file an issue first (or find an existing one) and announce that you plan to work on something. That way we will avoid having several people doing double work, and you might get useful feedback on the issue before you start working.
2931

3032
Browse through [`ARCHITECTURE.md`](ARCHITECTURE.md) to get a sense of how all pieces connects.
3133

@@ -34,76 +36,115 @@ You can test your code locally by running `./scripts/check.sh`.
3436
When you have something that works, open a draft PR. You may get some helpful feedback early!
3537
When you feel the PR is ready to go, do a self-review of the code, and then open it for review.
3638

37-
Please keep pull requests small and focused.
38-
3939
Don't worry about having many small commits in the PR - they will be squashed to one commit once merged.
4040

41-
Do not include the `.js` and `.wasm` build artifacts generated for building for web.
42-
`git` is not great at storing large files like these, so we only commit a new web demo after a new egui release.
41+
Please keep pull requests small and focused. The smaller it is, the more likely it is to get merged.
42+
43+
## PR review
44+
45+
Most PR reviews are done by me, Emil, but I very much appreciate any help I can get reviewing PRs!
4346

47+
It is very easy to add complexity to a project, but remember that each line of code added is code that needs to be maintained in perpituity, so we have a high bar on what get merged!
48+
49+
When reviewing, we look for:
50+
* The PR title and description should be helpful
51+
* Breaking changes are documented in the PR description
52+
* The code should be readable
53+
* The code should have helpful docstrings
54+
* The code should follow the [Code Style](CONTRIBUTING.md#code-style)
55+
56+
Note that each new egui release have some breaking changes, so we don't mind having a few of those in a PR. Of course, we still try to avoid them if we can, and if we can't we try to first deprecate old code using the `#[deprecated]` attribute.
4457

4558
## Creating an integration for egui
4659

47-
If you make an integration for `egui` for some engine or renderer, please share it with the world!
48-
I will add a link to it from the `egui` README.md so others can easily find it.
60+
See <https://docs.rs/egui/latest/egui/#integrating-with-egui> for how to write your own egui integration.
4961

50-
Read the section on integrations at <https://github.com/emilk/egui#integrations>.
62+
If you make an integration for `egui` for some engine or renderer, please share it with the world!
63+
Make a PR to add it as a link to [`README.md`](README.md#integrations) so others can easily find it.
5164

5265

5366
## Testing the web viewer
54-
* Install some tools with `scripts/setup_web.sh`
5567
* Build with `scripts/build_demo_web.sh`
5668
* Host with `scripts/start_server.sh`
5769
* Open <http://localhost:8888/index.html>
5870

5971

60-
## Code Conventions
61-
Conventions unless otherwise specified:
62-
63-
* angles are in radians and clock-wise
64-
* `Vec2::X` is right and `Vec2::Y` is down.
65-
* `Pos2::ZERO` is left top.
66-
72+
## Code Style
6773
While using an immediate mode gui is simple, implementing one is a lot more tricky. There are many subtle corner-case you need to think through. The `egui` source code is a bit messy, partially because it is still evolving.
6874

69-
* Read some code before writing your own.
70-
* Follow the `egui` code style.
71-
* Add blank lines around all `fn`, `struct`, `enum`, etc.
72-
* `// Comment like this.` and not `//like this`.
73-
* Use `TODO` instead of `FIXME`.
74-
* Add your github handle to the `TODO`:s you write, e.g: `TODO(emilk): clean this up`.
75-
* Write idiomatic rust.
76-
* Avoid `unsafe`.
77-
* Avoid code that can cause panics.
78-
* Use good names for everything.
79-
* Add docstrings to types, `struct` fields and all `pub fn`.
80-
* Add some example code (doc-tests).
81-
* Before making a function longer, consider adding a helper function.
82-
* If you are only using it in one function, put the `use` statement in that function. This improves locality, making it easier to read and move the code.
83-
* When importing a `trait` to use it's trait methods, do this: `use Trait as _;`. That lets the reader know why you imported it, even though it seems unused.
84-
* Follow the [Rust API Guidelines](https://rust-lang.github.io/api-guidelines/).
85-
* Break the above rules when it makes sense.
75+
* Read some code before writing your own
76+
* Leave the code cleaner than how you found it
77+
* Write idiomatic rust
78+
* Follow the [Rust API Guidelines](https://rust-lang.github.io/api-guidelines/)
79+
* Add blank lines around all `fn`, `struct`, `enum`, etc
80+
* `// Comment like this.` and not `//like this`
81+
* Use `TODO` instead of `FIXME`
82+
* Add your github handle to the `TODO`:s you write, e.g: `TODO(emilk): clean this up`
83+
* Avoid `unsafe`
84+
* Avoid `unwrap` and any other code that can cause panics
85+
* Use good names for everything
86+
* Add docstrings to types, `struct` fields and all `pub fn`
87+
* Add some example code (doc-tests)
88+
* Before making a function longer, consider adding a helper function
89+
* If you are only using it in one function, put the `use` statement in that function. This improves locality, making it easier to read and move the code
90+
* When importing a `trait` to use it's trait methods, do this: `use Trait as _;`. That lets the reader know why you imported it, even though it seems unused
91+
* Avoid double negatives
92+
* Flip `if !condition {} else {}`
93+
* Sets of things should be lexicographically sorted (e.g. crate dependencies in `Cargo.toml`)
94+
* Break the above rules when it makes sense
8695

8796

8897
### Good:
8998
``` rust
9099
/// The name of the thing.
91-
fn name(&self) -> &str {
100+
pub fn name(&self) -> &str {
92101
&self.name
93102
}
94103

95104
fn foo(&self) {
96-
// TODO(emilk): implement
105+
// TODO(emilk): this can be optimized
97106
}
98107
```
99108

100109
### Bad:
101110
``` rust
102-
//some function
103-
fn get_name(&self) -> &str {
111+
//gets the name
112+
pub fn get_name(&self) -> &str {
104113
&self.name
105114
}
106115
fn foo(&self) {
107-
//FIXME: implement
116+
//FIXME: this can be optimized
108117
}
109118
```
119+
120+
### Coordinate system
121+
The left-top corner of the screen is `(0.0, 0.0)`,
122+
with `Vec2::X` increasing to the right and `Vec2::Y` increasing downwards.
123+
124+
`egui` uses logical _points_ as its coordinate system.
125+
Those related to physical _pixels_ by the `pixels_per_point` scale factor.
126+
For example, a high-dpi screeen can have `pixels_per_point = 2.0`,
127+
meaning there are two physical screen pixels for each logical point.
128+
129+
Angles are in radians, and are measured clockwise from the X-axis, which has angle=0.
130+
131+
132+
### Avoid `unwrap`, `expect` etc.
133+
The code should never panic or crash, which means that any instance of `unwrap` or `expect` is a potential time-bomb. Even if you structured your code to make them impossible, any reader will have to read the code very carefully to prove to themselves that an `unwrap` won't panic. Often you can instead rewrite your code so as to avoid it. The same goes for indexing into a slice (which will panic on out-of-bounds) - it is often preferable to use `.get()`.
134+
135+
For instance:
136+
137+
``` rust
138+
let first = if vec.is_empty() {
139+
return;
140+
} else {
141+
vec[0]
142+
};
143+
```
144+
can be better written as:
145+
146+
``` rust
147+
let Some(first) = vec.first() else {
148+
return;
149+
};
150+
```

0 commit comments

Comments
 (0)