Breaking up Code in Unity (Important game dev tips for beginners)

  Рет қаралды 128,182

Lost Relic Games

Lost Relic Games

Күн бұрын

Пікірлер: 256
@pt8306
@pt8306 3 жыл бұрын
While your approach is certainly better than doing it all in one file, it still has a lot of dependencies. That is, the movement script still needs to know about the player script AND the input script in order to move. In some ways, you have actually made the dependencies worse, as everything now relies on the player script as well as any script that they would have relied on had you just referenced them directly. The only difference is they are getting it through the player script rather than accessing it directly. While this does have some positive implications for inheritence, as you can now specify subclasses in the player script and have your other script use those subclasses instead, there are still (in my opinion at least) better ways to do this. I would recommend making your scripts standard C# classes, rather than monobehaviours. This will lose your ability to drag and drop, but the benefits of having access to the constructor are worth it. Then, I would recommend passing in any needed dependencies through said constructors, creating (and passing) whatever is needed in the PlayerScript's Awake() function. For example, if the movementScript needs a character controller to move, it would be passed in as a constructor argument and assigned to a private variable within that class. By using this method, you can have your movement functions completely independent from input (and other scripts) entirely. This is good for things like code re-use, as your movement controller can now be used completely independently of your input source. The player script would be responsible for telling the objects how to move (by calling relevant functions in the movement controller) based on some events from the input controller. This way input can simply send events, no questions asked, the player can catch those events, and trigger movements based on them. Your input system can then be reused by any other monobehaviour that needs to capture input events, and your movement system can be reused for any charactercontroller independently of input (such as moving characters as part of a cutscene). You can even use interfaces or abstract classes here to completely overhaul functionality, something you can't do using monohebaviours since the unity inspector doesn't support interfaces at all. The only drawback is you lose drag and drop and instead have to create the other classes manually as part of your player script, but I feel the benefits are absolutely worth it, especially since you never have to worry about script execution order or any other unity shenanigans, and your object component heirarchy doesn't get bogged down with monobehaviours which really only exist to implement functionality and don't have/need options (they could all be set on the player component instead and passed to systems as needed instead).
@lonkk1
@lonkk1 Жыл бұрын
As someone who doesn’t use Unity, but works as a swe. This solution is much more comforting than the drag and drop.
@NeonValleys
@NeonValleys Жыл бұрын
This feels like a waste of time, at least for me. You have one example of it being useful, that of cutscenes, where you would for some reason need to move the NPCs exactly like the player. But that is super niche. And I can't think of any scenario that it would be useful to have my player scripts specifically undependant on each other and not monobehaviour.... It just doesn't make sense. For other things sure I can see the benefits immediately, but not this...
@pt8306
@pt8306 Жыл бұрын
@@NeonValleys Is it super niche though? If your movement code is separate from your input code, then you can re-use it for every character controller in the game - meaning your player(s) and your NPCs can all move in exactly the same way using exactly the same code - in the player's case, the movement is driven by inputs, and in the NPCs case, by AI, but in both cases they can use the same MovementController class to handle the actual movement, resulting in far less re-implementation of movement across multiple systems. Using the model in the original video (AND the model commonly used by a lot of Unity tutorials), you would have to code movement twice - once for the player and once for NPCs. The reason I mentioned cutscenes is because this is a less obvious benefit - not only can you control NPCs and the Player using the same code, but you can essentially turn the player into an NPC and bypass input entirely when you need to, such as in the cutscene example. If you can't see the instant benefits of this, I don't know what to tell you.
@NeonValleys
@NeonValleys Жыл бұрын
@@pt8306 I think so yes. It's only a very particular kind of game where you would even want the player and NPCs moving in the same way to begin with. And losing the ability to use it as a monobehaviour isn't worth it for something like the players movement and NPCs movement, especially for newer people learning from tutorials. It's absolutely useful to learn about using standard classes and interdependent code but not for something like simple player movement tutorials.
@itsbinxbro
@itsbinxbro 11 ай бұрын
@@NeonValleysIt's the same if you wanted things to have health in your game. Your player has health and can take damage, but so can an enemy, an explosive barrel, etc. You are going to want the health system to be the same across all these objects, but not have to write the same health code each time you create a new class. That's where this inheritance method thrives. When you want a mechanic that works not just for one object, but many different kinds of objects, you create a c# script that is modular, and can be attached to any object. But you can't do this if your scripts all rely on one-another to function. There are lots of methods of doing this and I'm still learning to implement them myself -- but in short: modular, decoupled scripts are crucial to a game project's longevity and scalability.
@devyoung4049
@devyoung4049 4 жыл бұрын
I attempted to break up my code following your video and failed the first time, 2 hours of work and I almost gave up.... But I kept trying and actually did it, I have clean code!!! I learned a lot and I'm inspired learn and code more. Thank you :)
@LostRelicGames
@LostRelicGames 4 жыл бұрын
That's epic! These things can be fiddly to setup the first time, but once they are in place they are a good foundation. Well don, all the best with your next steps on this journey!
@jeangodecoster
@jeangodecoster 4 жыл бұрын
For the record, whilst messing with script execution order IS a solution, it's a terrible solution. Mostly because it's something you don't see in the code, so problems linked to execution order are horrible to debug. And if someone else goes and mess with execution order (granted, that kind of problem doesn't happen in a solo project) you're up for a headache. GetComponent references should always be done in Awake, and pretty much any other case where you'd want to control the order in which some things happen can be dealt with in code with an Observer pattern.
@ArcangelZero7
@ArcangelZero7 4 жыл бұрын
"That kind of problem doesn't happen in a solo project." Ho-ho-ho my friend, you greatly underestimate my ability to sabotage myself!
@v4mp1r384
@v4mp1r384 4 жыл бұрын
@@ArcangelZero7 indeed, my past self and present self are totally different people. Future self is the one getting screwed the most.
@jeangodecoster
@jeangodecoster 3 жыл бұрын
@level90s still not a good practice, as you can’t use variables inside of your attributes, and attributes are not debuggable anyways. But hey at the end of the day it’s just a matter of good vs bad practice and if it works for you and the people who work with you, nothing is forbidden
@matejzajacik8496
@matejzajacik8496 2 жыл бұрын
Or, you can also go the normal sane way and simply have FULL control over the order of initialization, execution, life, and destruction of your objects. Relying on awakes, starts, and updates in individual components of individual game objects is needless and slow. To do this, you only have one Start() and one Update() in your whole project. In the Start(), you initialize everything, depending on any arbitrary conditions required and in the exact order as required. And the Update() becomes your main game loop, from where you call any other systems and entities and whatever, again, in the exact order and with the exact conditions your game requires. Heck, even Unity themselves preach this.
@RomainDelmaire
@RomainDelmaire Жыл бұрын
Even on a solo project, I imagine it would get really messy to maintain if you have to manually keep track of the order execution of each script.
@jampotjuh2465
@jampotjuh2465 3 жыл бұрын
I'm glad I found out about this before getting too far into my project
@muhammadhilmi2044
@muhammadhilmi2044 3 жыл бұрын
Dude, I just rage deleted my project beccause It's really messy...
@mayankbhaisora2699
@mayankbhaisora2699 3 жыл бұрын
@@muhammadhilmi2044 haha
@andersencastaneda6080
@andersencastaneda6080 3 жыл бұрын
Very interesting video, but I think you are making a circular reference. I suggest for decoupled and performant code: 1. Store your input data on a scriptable object, and then reference only that scriptable object in your movement script. This way your InputClass writes data to the scriptable object and your MovementClass reads the data inside your scriptable object. 2. Try to not use an update() on each of your scripts, instead, call your function OnUpdate() and create another script (UpdateManager) that handles all of your scripts that need to run on Update(). For Unity it's better to handle one Update() that do a thousand of things that a thousand Updates() that do one thing.
@pokoro97
@pokoro97 2 жыл бұрын
@@unitydev457 It would make sense that its faster since Start(), Update(), FixedUpdate(), etc in monobehaviours utilize the messaging system I believe, which is slower. Though by how much? You'd have to profile and see for yourself if it's worth it for your specific device and circumstances
@matejzajacik8496
@matejzajacik8496 2 жыл бұрын
@@pokoro97 In very small games, you will not see any difference. In a game with 2000+ game objects, each having 4-5 components, each having an Update() function, this becomes a thing, clearly visible in the profiler. The reason is that these "magical" functions are invoked in C# from the native code, and the marshalling is slow. Invoking a single C# Update() function in the whole project, and then from there calling whatever needs to happen within C# is way cheaper. What Andersen suggests makes perfect sense, of course. Not only do you gain performance-wise, you also have full control over the order of things, in a very natural way. The single Update() function becomes your main game loop, and then nothing prevents you from calling whatever code in whatever order with whatever conditions your game requires at any point. This is pretty cumbersome and error-prone via individual Updates().
@nonododo69
@nonododo69 2 жыл бұрын
1. It is not a good idea, because the scriptable objects stores data outside of play mode. So if you change value in the scriptable object then the file gets modified. So version controllers will see changes in the file every time you enter play mode. 2. Unity update works with reflection but it stores it. It is true you can get some plus performance if you have 1000+ objects with updates, but other way it doesn't really matter. We don't need to optimize everything, but just there where we have performance issues. (For loop always better in performance, but we use lists as well)
@sabeelm9
@sabeelm9 7 ай бұрын
​@nonododo69 I heard about this, someone once showcased scriptable objects behave differently when using the unity environenment vs actual deployment on a platform when the read / write privlages actually kicks in.
@Jazuhero
@Jazuhero 4 жыл бұрын
Good job highlighting an important topic for making your Unity code more manageable! There's also a third method, which I prefer, of controlling the execution order of Start() functions of parent-child scripts such as those shown in this video. In this method you leave the parent script's start code in the Start() function, but rename the child scripts' private Start() functions as something else, such as public CustomStart() so that Unity will no longer call them automatically. Just remember to make the custom start functions public so you can call it from another script. Then, you can call the CustomStart() functions manually in your parent script's default Start() function. This way you have total control over the script execution order, and can even have the custom start functions called in the middle of the parent Start() function, exactly as you deem necessary.
@kunalsamad2282
@kunalsamad2282 3 жыл бұрын
Thank you very much for this suggestion. As a beginner it is very helpful for me.
@sweeepeeee
@sweeepeeee 2 жыл бұрын
what the heck... if you don't want unity to automatically call Start, just don't tag this method. Incredible, right? I understand that you received information about Init, but something clearly went wrong)
@Jazuhero
@Jazuhero 2 жыл бұрын
@@sweeepeeee The point here was that you DO want Unity to call Start(), but only for one object. That object then calls the other objects' methods that would otherwise have also been Start(). This way all these methods get called at Unity's "Start() time", but you get to control their order.
@sweeepeeee
@sweeepeeee 2 жыл бұрын
@@Jazuhero yes, it's INITIALIZATION
@Lucas-hh4oh
@Lucas-hh4oh 3 жыл бұрын
Clean code is actually undervalued in indie game dev. Having object-oriented analysis and design skills really helps you to build the game and it will save you from hours of work.
@philippebaillargeon5204
@philippebaillargeon5204 Жыл бұрын
Cause a big part of Unity community aren't developers. They are video game enthusiasts. They don't know what is clean coding, separation of concerns, DRY pattern, encapsulation, etc.
@Alieldinofficial
@Alieldinofficial Жыл бұрын
I've seen games with 'bad' coding that could easily surpass games with 'good' coding. Game development is all about making a *good* game.
@popogames71
@popogames71 Жыл бұрын
​@@Alieldinofficial Yeah, but it is not about how good the game. Its about the behind the scense of the game. Without clean and well designed code- your game would not scale. Your bugs will require much more hourse of your time, you will not be able to reuse functionality you already wrote, it will be way harder for new people to understand what you did. I never understand why so many people insist on defending factually bad work habbits
@gimpdoctor8362
@gimpdoctor8362 3 жыл бұрын
I'm sorry if I'm wrong because I'm a beginner still, but it seems a better way of ordering these particular scripts' executions would be the following: - Disable the input/movement/collision scripts by unchecking their boxes in the inspector. - Enabling them within the player script's Start() function by using playerMovementScript.enabled = true; playerInputScript.enabled = true; etc. This way the Start() functions of each script can be manually run and code could even be run in between if needed. Also it's much clearer what's going on because you don't have to use the unity settings window to figure it out. Lastly ,for me and assumedly all beginners (and probably long time unity users too), the way in which Awake(), Start(), and Onenable() act is very confusing and difficult to remember, so here's a useful pair of links I'm constantly going back to to help remind myself. docs.unity3d.com/Manual/ExecutionOrder.html answers.unity.com/questions/372752/does-finction-start-or-awake-run-when-the-object-o.html (this one's more confusing even) answers.unity.com/questions/217941/onenable-awake-start-order.html
@LostRelicGames
@LostRelicGames 4 жыл бұрын
KZbin 3 hrs after upload decided to break the Audio Sync on the video. I'm just re-uploading a new version that will be linked here. Sorry for any inconvenience.
@CreepyUncleIdjit
@CreepyUncleIdjit 4 жыл бұрын
I thought my phone had a stroke while I was watching the video. Thanks for the update!
@LostRelicGames
@LostRelicGames 4 жыл бұрын
@@CreepyUncleIdjit It looks like they it fixed automatically! Would you mind checking for me and confirming if it's fixed?
@jaydev0860
@jaydev0860 4 жыл бұрын
@@LostRelicGames Looks fine to me now, and thanks for the video!
@CreepyUncleIdjit
@CreepyUncleIdjit 4 жыл бұрын
@@LostRelicGames Audio sync is good!
@BlueGooGames
@BlueGooGames 4 жыл бұрын
Yep works fine for me but I’m on mobile so I hade to watch real close :) great video, even for not being a total beginner :)
@diliupg
@diliupg 4 жыл бұрын
Excellent tutorial. Never knew we could set the execution order in the editor like that. Thanks a mil!
@LostRelicGames
@LostRelicGames 4 жыл бұрын
My pleasure man!
@coobbyo
@coobbyo 4 жыл бұрын
I just discovered your videos and like most people here I'm so grateful you put so much emphasis on quality. While there are channels out there that use concepts of quality code, very few teach them. Thank you so much. I can't wait to learn more!
@matejzajacik8496
@matejzajacik8496 2 жыл бұрын
Only what he teaches has nothing to do with quality code. It's nonsensical code with no benefits.
@saultoons
@saultoons 4 жыл бұрын
Great tip! I've started my own game recently and have heard a lot about splitting code up into different classes like this but wasn't sure how it would be done. This video helped a lot! More like this please, and post on reddit - gamedev or unity or whatever I'm sure others would find this useful :)
@LostRelicGames
@LostRelicGames 4 жыл бұрын
Really glad you liked it! I think it has been a difficult area for unity users due to how the component system works. But the benefits of separation is worth the work to make it happen! Completed sample project linked in the video description if you needed a closer look! :)
@saultoons
@saultoons 4 жыл бұрын
@@LostRelicGames Top man, cheers!
@bedtimestories1065
@bedtimestories1065 3 жыл бұрын
I took this a step further. There are 4 components coupled together here. One method that worked for me was creating super generic components. Specifically, I have two components. "SimpleMotor" and "CircleCaster". Either of those can be attached to any game object. SimpleMotor moves the object every FixedUpdate based on its MoveDirection. CircleCaster performs a Physics2D.CircleCast and invokes a C# delegate when something is hit. Now, you can create your more player-specific script and just set the motor's MoveDirection when input occurs and a collision occurs. Or, you can create AI that is totally independent of a PlayerInput scripts. It has worked VERY well for my project.
@tompainter7167
@tompainter7167 9 ай бұрын
For states, it is better to use enums than strings, is better Performance, prevents misspellings, better auto-complete and debug too
@mikelongchen7723
@mikelongchen7723 4 жыл бұрын
Thank you. I learned to program not too long and the term of pumping everything into a single class was deemed as the creation of the blob God. Now knowing how to separate the player controller, I will try and write better code.
@rashidfarhan6223
@rashidfarhan6223 3 жыл бұрын
the comment section is really educational and helpful! Cheers!
@PeterMilko
@PeterMilko 3 жыл бұрын
You can also put the scripts on children and parent always fires first, i think. So i would make a child object called "Collider" and put the script on there etc.
@_ZEG
@_ZEG 4 жыл бұрын
Thanks a lot man! I am a programming noob myself and often run into problems because I don't exactly understand yet how I'm supposed to structure my code. Your video made that much more clear!
@LostRelicGames
@LostRelicGames 4 жыл бұрын
I'm happy to hear that Zeg! Have a nice day :)
@dankingswell7082
@dankingswell7082 4 жыл бұрын
Your videos are so helpful my dude! moving from a web developer into game development it drives me mad how little information there is on good structure and best practices thanks again! :D
@marcossuel914
@marcossuel914 4 жыл бұрын
i've been looking for this for so long, thank youuu :)
@LostRelicGames
@LostRelicGames 4 жыл бұрын
I Hope you enjoy this! I truly appreciate each like and comment, they really help get this educational material in front of more people.
@BluePhantomStudio
@BluePhantomStudio 4 жыл бұрын
Question: When should we use this method
@dontownsend7748
@dontownsend7748 3 жыл бұрын
Very cool, didn't really understand the internal access modifier before this. Well-explained!
@marceldanz4436
@marceldanz4436 4 жыл бұрын
This is a very important concept to learn. The only thing I'm not liking quite in your code is that you have references in both directions. So Player Script knows of the Movement Script and Movement Script knows of the Player. It would be cleaner to have it only in one direction. Called a layered architecture. And components only grab things in layers below them, not above them. So for example the Movement script is in the top layer and the player Script in the one below. The communication between Movement and Input would then work that both change/use the Player Scripts state (making it a Model and having a clean MVC architecture). Regarding what you are calling spaghetti code: this is only a problem when you do it manually. This is where controllers come into play. You create a player controller that injects all the dependencies a subscript needs. So for example Movement Script uses Input Script. Then the Player controller injects this dependency (Input script) into a setup method of Movement script. I know I'm biting myself a bit with mixing layered arch and MVC, but my point is let's try to only use unilateral dependencies. :D Sorry this got long and hope you could follow me at least a part of the way. Great video though.
@DarthStevenus
@DarthStevenus 3 жыл бұрын
I'm confused, at 7:30 we see the PlayerMovementScript is calling the PlayerInputScript, albeit indirectly. I thought we wanted to eliminate the back and forth between them? Wouldn't it make more sense to have some public methods on the input script like IsLeftPressed() and IsRightPressed() which simply do something like "return Input.GetKey(KeyCode.D)"? Then expose some methods on PlayerMovementScript like MoveLeft() and MoveRight(). Then inside the PlayerScript Update method we could use the methods on the input controller to check for input, and if they return true call the movement controller methods? That way the input and movement controllers aren't communicating at all.
@VSalmerV
@VSalmerV Жыл бұрын
In my opinion script execution order is good only for attaching libraries and some extensions. Better way for this kind of problem is to create in root "PlayerScript" FlowManager that will have Awake, Start, Update, ect. methods and have a custom order of IUpdatable and so on components within. It will give a nice order control and will be opened for any extension you need later on
@smashies118
@smashies118 4 жыл бұрын
How did you directly answer my question about script structure the day I thought to ask it? Awesome! You confirmed my thought process isn't crazy and stupid.
@LostRelicGames
@LostRelicGames 4 жыл бұрын
haha! Glad to hear it Evan! A lot of things in unity feel unorthodox at times ;)
@ForTheOmnissiah
@ForTheOmnissiah 11 ай бұрын
Just for some clarity, the values in Execution Order are arbitrary and don't represent time passed. You can use any values that you see fit, as long as they values are in the order you want. You could use 1, 2, 3, or 100, 150, 200, or whatever. It simply executes them in numerical order from least-first to greatest-last, and the gap between values doesn't really make any difference.
@Влад-ъ4б6ц
@Влад-ъ4б6ц 2 жыл бұрын
The question is: how expensive this performance-wise? Because there are a lot of reaching to another scripts variables.
@MorRochben
@MorRochben 4 жыл бұрын
The best part of this practice is that you can reuse components like for example movement if your player controller and an enemy controller inherit from a unit controller script so you can have a unit controller field with either a enemy or player controller script in it. Now both your player and your enemy will be using the same movement script but they'll be called by an input script for your player and an AI script for you enemy, aka you have a bunch less duplicate code.
@DeathxStrike18
@DeathxStrike18 2 жыл бұрын
In your dry section you could have also allocated the rigid body as a private lambda as Private _playerRigidbody => playerScript.rb2d.velocity; in this case the _playerRigidbody is acting as a delegate stating when this value is used I need you to pass information to or from this path, in this case it does not matter when the Awake/start functions run and you can pass the new vector 2 all day.
@ingvmbv
@ingvmbv 2 жыл бұрын
Pretty handy video, I just started having same situation with many scripts, this is a good approach to be followed up
@jeangodecoster
@jeangodecoster 4 жыл бұрын
Hey great vids, youtube was absolutely right in suggesting you to me :-) here's an immediate sub. One interesting other approach to take is to have one component (playerscript) internally reference interfaces (IInputController, IHealth, IMover) which are pure C# interfaces. This allows to use the PlayerScript component as entry point for game loop callbacks, and hold global configuration, and have the logic happen in dedicated implementations, which you can adapt and even interchange if the game mechanics calls for it. It's a different approach, obviously, which may or may not be better depending on the use case, but it has the advantage to make for less granularity in your components, which once an object starts holding lots of logic, helps overall readability and usability.
@pt8306
@pt8306 3 жыл бұрын
I do this in my game, but with an extra layer. I have "entities", which are effectively pure C# game objects with built-in functionality (such as an inventory with AddItem, RemoveItem functions etc), then I have "controllers" which interact with these entities based on things like game state, then I have my MonoBehaviors which create their own instances of controllers, passing in any entity references as needed (sometimes entities are ScriptableObjects, sometimes not). This allows complete control (and replacement) of things like game rules while keeping object interactions not dependent on each other for functionality. This video means well, but this approach actually increases dependencies rather than reducing them, and will cause issues longer term for many people. It's better than doing everything in one file, but still has it's own share of issues.
@ilypavan
@ilypavan Жыл бұрын
After completing my project at last now when I decided to add sounds and sfx to my game everything is just breaking apart.
@wesleyCazelli
@wesleyCazelli 4 жыл бұрын
Instead of referencing the other scripts we could had used it inheritace right? Using inheritance in that case would provide a gain or loss in perfomance? ou none difference at all? BTW, great video, thanks.
@jeangodecoster
@jeangodecoster 4 жыл бұрын
Inheritance does not have an impact in performance, however Inheritance is the last resort in OO programming. Composition is more important than inheritance.
@gadgetboyplaysmc
@gadgetboyplaysmc 4 жыл бұрын
Why inheritance? It doesn't make sense. Are you saying you want the 3 other scripts (Input, Movement, Collision) to inherit from manager (PlayerScript)?
@wesleyCazelli
@wesleyCazelli 4 жыл бұрын
At the time i believe i was thinking playerScript inherit from the others. But you are right, it doesn't make sense. But what i really wanted to know is if there are any methods differente than communication that will result in a gain of performace.
@CodeStew
@CodeStew 3 жыл бұрын
Wish I saw this earlier. Now I'm breaking up a player script and now dealing with the fallout of private variable dependencies.
@Prakaz
@Prakaz 4 жыл бұрын
I'm just beginning to learn unity and this video is totally relevant and insightful.. The unity option for code-run-order looks very useful. Thanks!
@LostRelicGames
@LostRelicGames 4 жыл бұрын
My pleasure Prakaz, wishing you the best on your exciting ahead!
@Prakaz
@Prakaz 4 жыл бұрын
@@LostRelicGames thank you sir 🙏
@MichaelGGarry
@MichaelGGarry Жыл бұрын
I try to avoid Awake and certainly Start as much as possible. I instead have OneTimeSetup (for Get/Add Component and the like) and Initialisation (resting variables etc that can be called between levels/restarting the game etc) functions that I call from whatever the root class is via its Awake function. I then have complete control over calling order and its obvious in code what that is. Also for separating out the functionality - this approach still leaves a lot of dependency. Maybe events would be a better option?
@harrysadlermusic
@harrysadlermusic 4 жыл бұрын
Putting the project files on github would be a lot nicer than a google drive zip. Thanks for this tutorial! :)
@LostRelicGames
@LostRelicGames 3 жыл бұрын
good idea! I'll do a sweep of these in the near future
@JimmyKadesch
@JimmyKadesch 4 жыл бұрын
great video dude! im pretty deep into my first game and I wish I had this guidance earlier haha.
@LostRelicGames
@LostRelicGames 4 жыл бұрын
Glad you found value in! Wishing you the best with your project
@stefano4993
@stefano4993 2 жыл бұрын
Interesting video! Was just wondering why you use the internal scope for your fields even though the script is in the main assembly?
@LostRelicGames
@LostRelicGames 2 жыл бұрын
Hi mate, this is for convenice. Using internal will hide the value from the property inspector, yet allow puplic access from other scripts. Saves one adding meta tags to from inspector. :)
@stefano4993
@stefano4993 2 жыл бұрын
@@LostRelicGames Ah I see thanks. Basically the opposite of the SerializeField attribute :) You could also put it as a property.
@musab3575
@musab3575 2 жыл бұрын
how can i use this way for unityEvents ? like if i have a collision detection script that fires an event , do i add another unity event on the Player script and link the two events and make the other scripts listen to the Player script event?
@techcodenet
@techcodenet 4 жыл бұрын
Can you clarify benefit (and cons) of manually defining references between components like that - over alternatives? For example alternative is for each of those components to get and cache reference to other components it needs to interact with? Although I would actually have this inverted (object dealing damage would handle collision trigger and check if other has HandlesDamage component) ... Following your example of checking collision on the Player side - component handling collisions can do something like: void Awake(){ Health health = transform.GetComponent() } private void OnTriggerEnter(Collider other) { // Anything that DealsDamage will have this object DealsDamage damage= other.GetComponent(); if (damage != null) { health.ReduceHealth( damage.damageAmount() ); } } Without tightly coupling it to specifically Player allows reusing same components (e.g.: Health, DealsCollisionDamage, HandlesDamage) also on Enemies/Buildings ...etc
@Chris-zb5nm
@Chris-zb5nm 2 жыл бұрын
Well of course separating the codes that are not related to each other is a good thing to make codes cleaner. But what do you do about the performance? Imagine having 1 unit that have 1 script for controlling its behaviors. In a scene you have 1000 of that unit which means 1000 times Update/LateUpdate/FixedUpdate and so on. Now you split the codes into 3 scripts. 1000x units * 3 scripts = 3000 scripts. I'm just saying this to see other aspects as well. Maybe I'm wrong.
@Tomatech
@Tomatech 3 жыл бұрын
I try to seperate the player into a player movement controller and an entity controller, so the entity one can be reused by enemies and NPCs, and the player controller listens to messages from the Input System’s Input Controller
@LostRelicGames
@LostRelicGames 4 жыл бұрын
I released a Unity Asset to help beginners make platformers- ​u3d.as/2eYe​ Wishlist my game - store.steampowered.com/app/1081830/Blood_And_Mead/
@mehcaqueen
@mehcaqueen 3 жыл бұрын
I am new and reading over the code you provided. Could you explain why in the player script isLeftPressed and isRightPressed is declared as private, but in the input script they are declared as internal. You make that distinction several times in the code, what is the point / rule of thumb to keep using this practice?
@ricciogiancarlo
@ricciogiancarlo 4 жыл бұрын
omg you started right, but you did it wrong... all you did was just to split the PlayerScript.cs into 4 different files, but they are still highly dependant each other. I mean you changed the player state from the collision script! in order to move your rgb you check the InputScript inside the MovementScript. At 8:37 you said "dispatch" but you did't use any advantages from an event system, which is the right solution to decouple the components.
@martinvanstein.youtube
@martinvanstein.youtube Жыл бұрын
Nice video .. events/delegates would be better ... but as it is for beginners this is a nice intro as to how t get things to work and a bit easier to understand . the most important thing is to create your games and just like with everything else you'll get better and find new ways to do things
@ToniCorvera
@ToniCorvera 4 жыл бұрын
I have mixed feelings about this approach. But before I delve into this, I'd suggest adding "[RequireComponent]" attributes to each script so that adding one of them to a GameObject automatically adds the rest. Declaring the references as internal hurts my eyes a bit, since in the context of a Unity script that's essentially making them public. I was going to suggest a slightly different alternative but I'd rather actually test it beforehand in case I'm missing something.
@digitalconsciousness
@digitalconsciousness 3 жыл бұрын
I was confused as to how the collisionscript doesn't just create a new instance of playerscript when it creates it, but I understand now that it all has to do with connecting the serialized fields to each other. How would this normally be done in C++ I wonder? Passing playercollision what it needs through parameters by reference so it can change things in the manager?
@kunalsamad2282
@kunalsamad2282 3 жыл бұрын
Thank you very much for this video. As a beginner i guess i write too much of spaghetti codes. Now my script management is better than before.
@sanketvaria9734
@sanketvaria9734 3 жыл бұрын
I learned about these problems the hard way.
@rumuco
@rumuco 2 жыл бұрын
This is really helpful as I'm trying to add new thinks to mechanics but as my code mix input, animations, states, movement, etc, is kinda hard to add new things, now I'm going to start again, separate every aspect and try to make it work that way
@toninotonnato7776
@toninotonnato7776 3 жыл бұрын
as 3rd possible solution to the null reference problem produced by the scripts firing time: could we declare the children scripts (input/movement/collision) Start method as a coroutine (IEnumerator Start() ) and use WaitUntil to wait for the PlayerScript to initialise its variables?
@protophase
@protophase 2 жыл бұрын
Can you please explain how to utilize these scripts? There are no tutorials that I find that use this sort of separation between scripts so I don't know how to call a function from another script. I'm VERY MUCH a beginner and I would like to use this just to keep things organized from the start.
@tomashaddad
@tomashaddad 2 жыл бұрын
I don't think this is actually good design. If you go and find official Unity demos I think that's the best place to learn how to write clean code. Some of them aren't great but IIRC the input system video with the cartoon Viking does something similar to this video but doesn't create the upwards dependency by having a player script reference in the delegates. It's a small demo so you should be able to learn from it.
@Chubzdoomer
@Chubzdoomer 2 жыл бұрын
There are a gazillion videos here on KZbin covering how to call functions from other scripts in Unity. Just search for "unity call function from another script" and you'll find loads of them. The most important part is that the function you want to call from another script should be public. If it's private or protected, then other scripts won't be able to call/access it.
@Tuligarnio
@Tuligarnio 2 жыл бұрын
This is another approach that is more modular and decoupled that may be useful for you. kzbin.info/www/bejne/o3u1lGyhgd15ic0
@mad-j9985
@mad-j9985 3 жыл бұрын
Single responsibility principle. First one of the SOLID principles. You're welcome.
@diyguild1327
@diyguild1327 3 жыл бұрын
Just curious why are you using internal access modifiers for the variables? Is there some reason that you'd want the scope limited to that assembly unit over just limiting the scope to the class with private?
@pauljoneseyboy9615
@pauljoneseyboy9615 4 жыл бұрын
This is a game changer. Literally
@twokidsinatrenchcoat3549
@twokidsinatrenchcoat3549 3 жыл бұрын
Really great advice! thanks. Time to rewrite all of my code :)
@plufmot
@plufmot 2 жыл бұрын
13:19 I didn't know this! I thought the order of the components in the inspector determined the execution order ahaha
@daichi_devs
@daichi_devs 4 жыл бұрын
Stupid question: is internal same as private?
@LostRelicGames
@LostRelicGames 4 жыл бұрын
Not stupid at all! Internal is sort of like private. it means that other classes in the inheritance chain can also access it.
@savecandy6881
@savecandy6881 4 жыл бұрын
lol I faced this problem but I used just awake function .. Thank you a lot for the new way
@LostRelicGames
@LostRelicGames 4 жыл бұрын
haha no problem! :) it's interesting how many ways we can solve the same problem in unity and programming in general.
@asanzramaxi
@asanzramaxi 3 жыл бұрын
light mode?
@j.j.maverick9252
@j.j.maverick9252 2 жыл бұрын
Is there a clean pattern for when movement has to embed collision as it does in tile based 2d (where we separate movex+collx from movey+colly)?
@RomainDelmaire
@RomainDelmaire Жыл бұрын
I'm a newb but so far, I've always heard that you should only mess with script execution order as an absolute last resort when you have absolutely no other option available. I can imagine that if a project gets larger, it would be an absolute nightmare to maintain, especially if multiple people are working on the same project.
@opensourcefreedom9241
@opensourcefreedom9241 4 жыл бұрын
Is there a way to use "using x" or maybe an "import"? Having done a lot of scripting in other languages there is always a way to make the hierarchy work, but I am new to C#. So basically I am wondering if you can include or use other scripts at the head of a main script so they can be used. I saw another good comment about making use of an interface. I would be interested to see if there are other good ways of implementing this so that it is more programmatic and less configuration based.
@markhenrikson9790
@markhenrikson9790 3 жыл бұрын
I am very new to programming and what experience I have is from long ago in Basic and Fortran... yes... longggggg ago! If I follow the discussion, when you talk about breaking up code... is this similar to earlier times when you would use structured programming and put routines that would be used repeatedly in a subroutine that could be called as needed or am I missing the point of this altogether?
@JackyTran
@JackyTran 3 жыл бұрын
Would code separation cause any slowness?
@Hozgen
@Hozgen Жыл бұрын
we can that it is mediator design or a kind of visitor (not directly)
@Critters
@Critters 3 жыл бұрын
The collision script, it has to know that the player script has a ChangeState function that is looking for "Player_happy" or "Player_sad". What if you wanted to use this collision script on an enemy character, or a bullet, or a crate? A follow-up video where you take this concept to the next level would be cool. There are a couple of different ways to skin this cat, maybe you could discuss the pros and cons of each.
@rmt3589
@rmt3589 Жыл бұрын
What's wrong with actually putting it all on one file?
@rmt3589
@rmt3589 Жыл бұрын
@@robbyz512 I have grown a lot since then. Forgive my grave ignorance.
@Monzi123456789
@Monzi123456789 4 жыл бұрын
@Lost Relic Games, Why not just make partial Classes in that case?
@Iboshido
@Iboshido 3 жыл бұрын
what's the advantage of using "internal" instead of "public"?
@LostRelicGames
@LostRelicGames 3 жыл бұрын
It makes it public in scope without revealing it in the inspector
@casachezdoom2588
@casachezdoom2588 3 жыл бұрын
By setting how many ms before a script runs, can't that potentially make your scripts run slower than they could since they have to wait a certain amount of time?
@TheLegendsOfTynedale
@TheLegendsOfTynedale 4 жыл бұрын
Great video! Help me: is script separation just about ease of read, or does it improve the performance of everything related to the object? For example, are collision detections weighted down by long scripts, leading to errors, or is it immaterial? Thank you!!
@jeangodecoster
@jeangodecoster 4 жыл бұрын
separation will always be less performant than having everything in one single component. This is neither about ease of read (well yeah partly but not essentially), nor about performance, it's about scalability and architecture (single responsibility principle). Referencing other components has a cost, although negligible (unless you do a GetComponent in update... always cache component references in Awake). It's never going to be more performant. The main problem of long scripts is indeed readability, but also changing one small thing may impact the rest of the code. That's why there's the Open/Closed principle and the Single Responsibility principle.
@gadgetboyplaysmc
@gadgetboyplaysmc 4 жыл бұрын
I don't really think it's about performance but more about having a good game architecture. But doing this doesn't bog down your performance in any way either. It makes it more organized rather than having a single monolithic class full of variables and if and else statements(which is extremely hard to manage). Which means it's easier to work with so you don't run into any problems in the future. And I emphasize on "The Future" because it means your code is going to be a lot easier to work with when you add more logic into it. Sure, you can get away with being very careful when handling your scripts because you are the developer but that's not gonna work in the long run especially if you work with other people. Sorry if that's too vague tho. I can't really think of an example right off the bat but script separation is a really simple way to help with just writing great code overall.
@TheLegendsOfTynedale
@TheLegendsOfTynedale 4 жыл бұрын
@jean-gobert de coster hey. Thanks for this. Yeah I can see the advantages.
@TheLegendsOfTynedale
@TheLegendsOfTynedale 4 жыл бұрын
@@gadgetboyplaysmc hey. Thanks. Definitely get what you mean so definitely not vague. Nice one
@javadasadi3267
@javadasadi3267 2 жыл бұрын
Thanks for the informations I wonder is there any way to assign all those three scripts ( movement, collision, input) to main player script via player script itself without doing them directly from inspector. And if so ,does it keeps the scripts order ? Again thanks for the video
@mrjeanjean6794
@mrjeanjean6794 4 жыл бұрын
Hi! Thanks for all your sharings. Got a quick question: In your architecture, there is some benefits of using SerializedField variables (so you need to drag and drop scripts instance like you did) instead of using a GetComponent method to get the references of the other scripts? Hope I'm clear. Thx
@dovahjaron
@dovahjaron 2 жыл бұрын
Hey I don't know if your still looking at these comments but I noticed in your code you often use internal as apposed to what I often see which is using public or private as a variable prefix. What is that doing exactly and how does that differ from the mentioned public/private?
@LostRelicGames
@LostRelicGames 2 жыл бұрын
Hi Dovah, in a nutshell, internal acts like public without exposing the variables in the inspector panel as public would.
@dovahjaron
@dovahjaron 2 жыл бұрын
@@LostRelicGames Oh okay that makes sense, thank you!
@ineeda12step92
@ineeda12step92 4 жыл бұрын
Very informative, thank you! Have you found any performative issues with separating the script other than the firing order? Conversely, have you found that this approach makes the project better performing? Thanks again for all of the time you put into these videos!
@LostRelicGames
@LostRelicGames 3 жыл бұрын
Hey mate! As the scripts are stored at startup there is no notable performance cost during runtime.
@ineeda12step92
@ineeda12step92 3 жыл бұрын
@@LostRelicGames awesome! Thank you for the feedback.
@Ferenc-Racz
@Ferenc-Racz 3 жыл бұрын
Hi! Thank you for your help / explanation. I did something similar in my project, but I had some problem when my "backend or business logic" needs some aditional information from the User/Ui. Could you give me some advice regarding this? (Example: 1) the player plays a card, which is an upgrade card 2) then the card logic needs an additional information which reaource the player wants upgrade.) I have a working solution for it with async tasks methods, but it looks ugly and hard to debug. Im sure there should be a better way, but I dont know it yet. Thanx in advance if you answer.
@m_maksym
@m_maksym 3 жыл бұрын
Just want to say thanks for nice and usefull tips! For me, as beginner in unity -they're just priceles))
@quantumdev6577
@quantumdev6577 2 жыл бұрын
Awesome topic! Thank you for explaing all of this very well. I really appreciate that.
@lAztechl
@lAztechl 4 жыл бұрын
Hi, Great Tutorial and I subscribe to your channel. Thank you. Can I ask a question about the separation of code you point out in the video? Like in the process if the children want to talk to other children they need to communicate first with the parent is that right? Or can I just like if the movement needs the input, I just listen to the event of input directly that can be set in the movement awake/start? And also the rule is all the reference component will be int the parent and the children cannot communicate directly in the other children? and the last question can, Is having a code like for example the parent have input and movement script, the input fire, the input pass the data in parent, for example, a left button and the parent have a logic to determine if what function in the movement to call, is this code okay? or I need only the parent for referencing the children?
@vast634
@vast634 3 жыл бұрын
You structure now depends on however you set up your components (correctly) in the editor. Why not just create simple classes, that get spawned by the player component, get a reference to the playerscript in their constructor and get processed every on update? Less error prone than adding and referencing separate Unity-components.
@oxirosmusic
@oxirosmusic 3 жыл бұрын
Coming from iOS dev this looks very weird lol. Could you just extend the class in different files? Or does delegation exists in unity?
@kirillkir6268
@kirillkir6268 4 жыл бұрын
Managing order of scripts loading in project settings looks like a spike. Have you ever use DI in Unity? It would be interesting to look best practice
@jeangodecoster
@jeangodecoster 4 жыл бұрын
Well, technically, the Component architecture is a form of DI There are DI frameworks for unity. Zenject is the most popular afaik. However it doesn't play well with Unity's general architecture, essentially you're putting a DI framework (classic DI) on top of another DI framework (Components). As for script execution ordering, actually this is bad design. Never reference other components in Start, ALWAYS do that in Awake
@saito853
@saito853 3 жыл бұрын
Not a fan of how you're doing it, the child scripts should never have to to depend on a different child script, with how you're doing it, removing one script means that the entire system is broken, because all the scripts are dependent on one another. If it was me, I'd further break it down to different interfaces containing necessary properties. For example in your diagram at 3:10 there are the Input, Movement and Collision scripts, I'd have interfaces for those that need to feed information back to the main script to ensure the parent script has the necessary properties. So for example I would have an IMovementProperties interface that contains value for movement related stuffs, and have the "Input" script modifies those property while the "Movement" script reads them for movement purposes. This way you can for example, disable the Input script for either gameplay or debug purposes, and still have the movement script running (to manipulate it with other means, such as an auto-pathing feature).
@anthonyrochet6858
@anthonyrochet6858 3 жыл бұрын
I inspired myself from the ECS stack, Entity, Component, System in order to architecture my project. Components hold the data for a very specific aspect of my game. They require the presence of their dependencies, mainly unity related components. - MouvementComponent (need the NavMeshAgent, a collider and a kinematic rigidbody in order to work. I not present, it add it to the gameObject.) - ContainerComponent (Hold the data of items, contain all data needed to manage a container like max quantity, current quantity present, etc...) Systems hold the logic for a very specific aspect of the game. I consider them as the API of each aspect of my game. They require the presence of their dependencies, mainly components, sometimes systems, or a specific entity. - MouvementSystem (need the MouvementComponent in order to work. I not present, it add it to the gameObject.) This system allow me to move the related gameObject within the bounds of the API I define myself, exposing only what I consider to be my domain API to the other gameObjects. - ContainerSystem (need the ContainerComponent in order to work.) This system allow me to stock items in gameObjects and manage transactions of items betweens ContainerSystems, enforcing security check. - ContainerStockTagSystem (need the ContainerComponent) Act as a notifier with a dedicated event. This tag this ContainerComponent as a "Stock" => A container where IA can store and collect stuff. For instance, inventory of a PNJ is a container but not a stock as other PNJ can't stock stuff inside if they need to. Entities hold the references of each components and systems that are needed for the expected behavior. If the components aren't presents, it add it. Same for the systems. They are used to interract between gameObjects. When two entities need to interract, say a Stock and a PNJ (StockEntity and TravelerEntity) they interract through their entities. If a component or a system is not exposed in the entity, it's not meant to be used by other entities and is private. The ContainerStockTagSystem example above illustrate this, it's a private system of my StockEntity. I hope it can help someone.
@matejzajacik8496
@matejzajacik8496 2 жыл бұрын
Jesus, let's complicate something that can be stupidly simple into a huge OOP castle in the clouds... Not a fan of your suggestion. Games are extremely complex and things simply HAVE TO depend on other things. Deal with it.
@greypo1886
@greypo1886 3 жыл бұрын
What I learned from my lags Do not separate variables that stores data for a method By making reference to another Class I think I should link scripts that only "invoke" methods As possible
@againstobs
@againstobs 3 жыл бұрын
Learnt so much thanks to you !
@TamashiiRyu
@TamashiiRyu 4 жыл бұрын
Is there a benefit to this method vs creating the Player/Game Controller as a static class?
@flaviokonti5522
@flaviokonti5522 4 жыл бұрын
Singletons have their uses and problems related to them. I would advise to investigate about them before you start using them on your projects
@pedrohenriquecesargodoi8336
@pedrohenriquecesargodoi8336 3 жыл бұрын
This is a bad practice, the best solution is using code Inheritance. You make each code dependent on the others, in your example PlayerInput will depend on PlayerScript, PlayerMovement will depende on PlayerInput (which also will depend on PlayerScript) and so on. Also using code inheritance you can make a base script for several other scripts. A "CharacterHealth" script can be based to be used for the player, enemies, player allies, etc.
@DINGOS30
@DINGOS30 2 жыл бұрын
I've never heard anyone talk about this problem. How are ppl not having this issue and speaking up about it!?
@BaconAndMinecraft
@BaconAndMinecraft 4 жыл бұрын
could I use inheritance in situations like these or a static class for the main player script maybe? I’ve gotten into programming in unity recently but I’m not really sure where to use all the different methods, maybe a future video? great tutorial btw, I like these raw videos
@xyzek0157
@xyzek0157 4 жыл бұрын
Wow, that is actually very useful. Im very impressed by your videos, they truly are beginner friendly. Also im suprised you dont have as much views as for example Brackeys which videos are in my oppinion much less useful than yours. Now that's an instant subscribe and Blood and Mead wishlist my friend! Also i have a question for you. Could this problem with start method execute order be solved with static method? I could be wrong but i think that static methods are called before non static.
@LostRelicGames
@LostRelicGames 4 жыл бұрын
Hey man, thanks so much for the kind words. KZbin has a strange algorithm thay often doesn't reward the small channels. I'll keep pushing on though! Thanks for the sub and wishlist. Come by the discord some time, great place for getting help and sharing progress, it would be a pleasure to have a person with your positive attitude part of the community. Regarding statics, they are sort of bad practice generally, and used more for global referencing.
@xyzek0157
@xyzek0157 4 жыл бұрын
@@LostRelicGames Thanks for the reply, ill surely join your discord server :)
@Jeamar
@Jeamar 4 жыл бұрын
Hey! I found this tutorial really helpful, but got me thinking: what do you think of using inheritance in unity? I.e.: a character class (that then has player and enemy childs) who manages the basics, like health, death and rigidbody basics. And how could you work both splitting the code and using inheritance?
@davonphillips8048
@davonphillips8048 4 жыл бұрын
Hey so I'm looking to develop my own game with unity. Is C# the only code you can use for unity or can you use other codes like python
@LostRelicGames
@LostRelicGames 4 жыл бұрын
C# is the primary language used. And its pretty cool! :)
@flaviokonti5522
@flaviokonti5522 4 жыл бұрын
If you want to use python, check Godot game engine, its pretty cool
@DukensteinA1
@DukensteinA1 4 жыл бұрын
Excellent tutorial!
@that_person436
@that_person436 3 жыл бұрын
This helped me a lot. Thank you
@dinogon5121
@dinogon5121 3 жыл бұрын
Many thanks. It help me so much
@daichi_devs
@daichi_devs 4 жыл бұрын
Would you also put the state script as a separate code?
@LostRelicGames
@LostRelicGames 4 жыл бұрын
I personally keep the state management in the main script, though if the state manager was particularly complicated , I may consider separating it.
@daichi_devs
@daichi_devs 4 жыл бұрын
Lost Relic Games thank your for your replies. You make awesome videos that really help out noobs like me, thank you so much.
@miriomandubisi4593
@miriomandubisi4593 3 жыл бұрын
Pls show is a tutorial on how we can receive payment from Unity
4 Ways to Code 2D Player Movement in Unity (You SHOULD know these!)
16:24
Lost Relic Games
Рет қаралды 172 М.
20 Advanced Coding Tips For Big Unity Projects
22:23
Tesseract
Рет қаралды 205 М.
小丑女COCO的审判。#天使 #小丑 #超人不会飞
00:53
超人不会飞
Рет қаралды 16 МЛН
Don’t Choose The Wrong Box 😱
00:41
Topper Guild
Рет қаралды 62 МЛН
Unity3D Managers vs Controllers
12:06
Jason Weimann (GameDev)
Рет қаралды 58 М.
Better Coding in Unity With Just a Few Lines of Code
15:27
Firemind
Рет қаралды 318 М.
Godot Scripts I add to Every Game
12:34
Aarimous
Рет қаралды 56 М.
10 Unity Tips You (Probably) Didn't Know About
6:47
Sasquatch B Studios
Рет қаралды 48 М.
I Scraped the Entire Steam Catalog, Here’s the Data
11:29
Newbie Indie Game Dev
Рет қаралды 576 М.
Why Solo Developers Should Use Unreal
9:51
Thomas Brush
Рет қаралды 456 М.
The Most Wishlisted Steam Game Just Exploded on Launch
9:43
Lost Relic Games
Рет қаралды 40 М.