• Log InLog In
  • Register
Liquid`
Team Liquid Liquipedia
EST 21:57
CET 03:57
KST 11:57
  • Home
  • Forum
  • Calendar
  • Streams
  • Liquipedia
  • Features
  • Store
  • EPT
  • TL+
  • StarCraft 2
  • Brood War
  • Smash
  • Heroes
  • Counter-Strike
  • Overwatch
  • Liquibet
  • Fantasy StarCraft
  • TLPD
  • StarCraft 2
  • Brood War
  • Blogs
Forum Sidebar
Events/Features
News
Featured News
RSL Season 3 - Playoffs Preview0RSL Season 3 - RO16 Groups C & D Preview0RSL Season 3 - RO16 Groups A & B Preview2TL.net Map Contest #21: Winners12Intel X Team Liquid Seoul event: Showmatches and Meet the Pros10
Community News
Weekly Cups (Nov 24-30): MaxPax, Clem, herO win2BGE Stara Zagora 2026 announced15[BSL21] Ro.16 Group Stage (C->B->A->D)4Weekly Cups (Nov 17-23): Solar, MaxPax, Clem win3RSL Season 3: RO16 results & RO8 bracket13
StarCraft 2
General
Chinese SC2 server to reopen; live all-star event in Hangzhou Maestros of the Game: Live Finals Preview (RO4) BGE Stara Zagora 2026 announced Weekly Cups (Nov 24-30): MaxPax, Clem, herO win SC2 Proleague Discontinued; SKT, KT, SGK, CJ disband
Tourneys
RSL Offline Finals Info - Dec 13 and 14! RSL Offline FInals Sea Duckling Open (Global, Bronze-Diamond) $5,000+ WardiTV 2025 Championship Constellation Cup - Main Event - Stellar Fest
Strategy
Custom Maps
Map Editor closed ?
External Content
Mutation # 502 Negative Reinforcement Mutation # 501 Price of Progress Mutation # 500 Fright night Mutation # 499 Chilling Adaptation
Brood War
General
BW General Discussion Which season is the best in ASL? Data analysis on 70 million replays BGH Auto Balance -> http://bghmmr.eu/ [ASL20] Ask the mapmakers — Drop your questions
Tourneys
[BSL21] RO16 Group D - Sunday 21:00 CET [BSL21] RO16 Group A - Saturday 21:00 CET [Megathread] Daily Proleagues [BSL21] RO16 Group B - Sunday 21:00 CET
Strategy
Current Meta Game Theory for Starcraft How to stay on top of macro? PvZ map balance
Other Games
General Games
ZeroSpace Megathread Stormgate/Frost Giant Megathread Nintendo Switch Thread The Perfect Game Path of Exile
Dota 2
Official 'what is Dota anymore' discussion
League of Legends
Heroes of the Storm
Simple Questions, Simple Answers Heroes of the Storm 2.0
Hearthstone
Deck construction bug Heroes of StarCraft mini-set
TL Mafia
Mafia Game Mode Feedback/Ideas TL Mafia Community Thread
Community
General
Things Aren’t Peaceful in Palestine Russo-Ukrainian War Thread US Politics Mega-thread The Big Programming Thread Artificial Intelligence Thread
Fan Clubs
White-Ra Fan Club
Media & Entertainment
[Manga] One Piece Movie Discussion! Anime Discussion Thread
Sports
2024 - 2026 Football Thread Formula 1 Discussion NBA General Discussion
World Cup 2022
Tech Support
Computer Build, Upgrade & Buying Resource Thread
TL Community
Where to ask questions and add stream? The Automated Ban List
Blogs
James Bond movies ranking - pa…
Topin
Esports Earnings: Bigger Pri…
TrAiDoS
Thanks for the RSL
Hildegard
Saturation point
Uldridge
Customize Sidebar...

Website Feedback

Closed Threads



Active: 1984 users

The Big Programming Thread - Page 490

Forum Index > General Forum
Post a Reply
Prev 1 488 489 490 491 492 1032 Next
Thread Rules
1. This is not a "do my homework for me" thread. If you have specific questions, ask, but don't post an assignment or homework problem and expect an exact solution.
2. No recruiting for your cockamamie projects (you won't replace facebook with 3 dudes you found on the internet and $20)
3. If you can't articulate why a language is bad, don't start slinging shit about it. Just remember that nothing is worse than making CSS IE6 compatible.
4. Use [code] tags to format code blocks.
Shield
Profile Blog Joined August 2009
Bulgaria4824 Posts
Last Edited: 2014-06-14 22:51:44
June 14 2014 22:48 GMT
#9781
Thanks! I have another question, this time regarding SRP (Single Responsibility Principle).

So if you have a class Person (standard fields such as name, age, etc). Is it reasonable to have a method, e.g. calculateAge() which calculates person's age based on their birth date? Basically, I'm confused if such a method is regarded as a second responsibility for the class, thus violation of SRP. So, how do you go with reasoning in such cases?

On the other hand, this reminds me of 'Feature Envy' (from the book Clean Code), so such a method would wish to be in the class perhaps. Anyway, I'm just confused as I said.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
Last Edited: 2014-06-14 23:11:41
June 14 2014 23:02 GMT
#9782
--- Nuked ---
WolfintheSheep
Profile Joined June 2011
Canada14127 Posts
June 14 2014 23:04 GMT
#9783
On June 15 2014 07:48 darkness wrote:
Thanks! I have another question, this time regarding SRP (Single Responsibility Principle).

So if you have a class Person (standard fields such as name, age, etc). Is it reasonable to have a method, e.g. calculateAge() which calculates person's age based on their birth date? Basically, I'm confused if such a method is regarded as a second responsibility for the class, thus violation of SRP. So, how do you go with reasoning in such cases?

On the other hand, this reminds me of 'Feature Envy' (from the book Clean Code), so such a method would wish to be in the class perhaps. Anyway, I'm just confused as I said.

The point of design principles isn't to create a rigid set of rules, they're to create sensible code structure.

In this specific case, you really have to ask "Should the age of the person be accessible from the Person class"? Or, maybe more importantly, "If I reuse this Person class, should I have to import a whole other class just to find the age"?
Average means I'm better than half of you.
Beef Noodles
Profile Blog Joined March 2010
United States937 Posts
Last Edited: 2014-06-14 23:55:17
June 14 2014 23:54 GMT
#9784
Quick question for anyone in the know: is there a trusted textbook for learning VBA? I am comfortable coding in python and java, but I want to pick up VBA. So far, I'm finding it hard to find a good book, because there are so many ads for books on excel and VBA geared toward finance types with no coding experience (and honestly they don't look very good).

I was just seeing if anyone has read a comprehensive VBA book they found particularly good.

Thanks!
spinesheath
Profile Blog Joined June 2009
Germany8679 Posts
June 15 2014 06:55 GMT
#9785
On June 15 2014 07:48 darkness wrote:
Thanks! I have another question, this time regarding SRP (Single Responsibility Principle).

So if you have a class Person (standard fields such as name, age, etc). Is it reasonable to have a method, e.g. calculateAge() which calculates person's age based on their birth date? Basically, I'm confused if such a method is regarded as a second responsibility for the class, thus violation of SRP. So, how do you go with reasoning in such cases?

On the other hand, this reminds me of 'Feature Envy' (from the book Clean Code), so such a method would wish to be in the class perhaps. Anyway, I'm just confused as I said.

"Single Responsibility Principle" is a somewhat misleading name. SRP means that there "should only be one reason for a class to change"; this is not necessarily the same as "should only do one thing".

Whether you "do one thing" or not changes easily as you move up or down levels of abstraction. For example you could say that your class provides access to a person's name AND age AND so on, or you could say that it provides access to person-related information.

That person class would have to change when you change the range of information it offers about a person. If it also would have to change when you change your data management system or various other things, then that violates the SRP.
If you have a good reason to disagree with the above, please tell me. Thank you.
CatNzHat
Profile Blog Joined February 2011
United States1599 Posts
Last Edited: 2014-06-15 09:26:25
June 15 2014 09:24 GMT
#9786
On June 15 2014 07:48 darkness wrote:
Thanks! I have another question, this time regarding SRP (Single Responsibility Principle).

So if you have a class Person (standard fields such as name, age, etc). Is it reasonable to have a method, e.g. calculateAge() which calculates person's age based on their birth date? Basically, I'm confused if such a method is regarded as a second responsibility for the class, thus violation of SRP. So, how do you go with reasoning in such cases?

On the other hand, this reminds me of 'Feature Envy' (from the book Clean Code), so such a method would wish to be in the class perhaps. Anyway, I'm just confused as I said.

The age accessor definitely belongs in the Person model. A one line method as a mixin that will only ever be able to be used in on a single person object would be silly. Maybe something slightly more complicated, such as a function for calculating the distance between two people would belong in a different class.
sluggaslamoo
Profile Blog Joined November 2009
Australia4494 Posts
Last Edited: 2014-06-15 10:23:22
June 15 2014 10:21 GMT
#9787
On June 15 2014 08:02 Nesserev wrote:
Show nested quote +
On June 15 2014 07:48 darkness wrote:
Thanks! I have another question, this time regarding SRP (Single Responsibility Principle).

So if you have a class Person (standard fields such as name, age, etc). Is it reasonable to have a method, e.g. calculateAge() which calculates person's age based on their birth date? Basically, I'm confused if such a method is regarded as a second responsibility for the class, thus violation of SRP. So, how do you go with reasoning in such cases?

On the other hand, this reminds me of 'Feature Envy' (from the book Clean Code), so such a method would wish to be in the class perhaps. Anyway, I'm just confused as I said.

If there's one thing that I learned about SRP, is that you don't want to overthink it... just keep a class as general and as easy to work with as possible, there are so many ways to look at a class, and say "well, I could split this up in ten different classes if I look at it in a different way".

In my opinion, it's okay to have a calculateAge()- or a getAge()-method for the Person class. If you look at Person as a class that governs all the information about a certain person, then just like birthdate and name, age is just another piece of information, even if you don't supplement it directly to the class, and need an extra function to calculate it.

Show nested quote +
On June 14 2014 10:30 mcc wrote:
As far as function size goes, some can be refactored, but in general sometimes large body just makes most sense, because there is no logical grouping of statements that could make a meaningful method.

I agree that duplicate code is sometimes preferable, mostly for readability; but having a large body of code almost always means that are multiple steps/phases, so you could definitely split that body up per phase/step into smaller functions.


You would have a function called Age() which would be called everytime you want to access a persons age. The performance impact should be negligible, however if you wanted to optimise you could use lazy loading.

As for method size, 99.999% of the time there is a way to split it up into smaller methods. If you know how to do it properly it will always make the code more maintainable and neater.
Come play Android Netrunner - http://www.teamliquid.net/forum/viewmessage.php?topic_id=409008
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
Last Edited: 2014-06-15 15:22:04
June 15 2014 14:42 GMT
#9788
--- Nuked ---
spinesheath
Profile Blog Joined June 2009
Germany8679 Posts
June 15 2014 14:50 GMT
#9789
On June 15 2014 23:42 Nesserev wrote:
Show nested quote +
On June 15 2014 19:21 sluggaslamoo wrote:
You would have a function called Age() which would be called everytime you want to access a persons age. The performance impact should be negligible, however if you wanted to optimise you could use lazy loading.

In my opinion, storing age is rather redundant (if you already have the birthdate), and potentially 'dangerous'.

Storing age might actually not be a good idea, since age is based on the difference between the current date and the date of birth, and the current date changes every day. Imagine if the age-variable of a person is initiated the day before his birthday... the next day his age won't be correct anymore. Of course, you can have mechanisms in place to keep track of whether or not the age is still correct, but that would totally negate the advantages of 'lazy evaluation'.

Nobody says that you would store age. If you already store the date of birth, you could calculate age based on a somehow obtained "now" value. Whether you can get your hands on such a "now" without causing any problems depends on various circumstances.
If you have a good reason to disagree with the above, please tell me. Thank you.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
Last Edited: 2014-06-15 15:23:24
June 15 2014 15:21 GMT
#9790
--- Nuked ---
mcc
Profile Joined October 2010
Czech Republic4646 Posts
June 15 2014 15:24 GMT
#9791
On June 16 2014 00:21 Nesserev wrote:
Show nested quote +
On June 15 2014 23:50 spinesheath wrote:
On June 15 2014 23:42 Nesserev wrote:
On June 15 2014 19:21 sluggaslamoo wrote:
You would have a function called Age() which would be called everytime you want to access a persons age. The performance impact should be negligible, however if you wanted to optimise you could use lazy loading.

In my opinion, storing age is rather redundant (if you already have the birthdate), and potentially 'dangerous'.

Storing age might actually not be a good idea, since age is based on the difference between the current date and the date of birth, and the current date changes every day. Imagine if the age-variable of a person is initiated the day before his birthday... the next day his age won't be correct anymore. Of course, you can have mechanisms in place to keep track of whether or not the age is still correct, but that would totally negate the advantages of 'lazy evaluation'.

Nobody says that you would store age. If you already store the date of birth, you could calculate age based on a somehow obtained "now" value. Whether you can get your hands on such a "now" without causing any problems depends on various circumstances.

Storing the value is part of 'lazy loading', which is what sluggaslamoo recommended as a optimization...
(I did use the term 'lazy evaluation' instead of 'lazy loading' in my answer... will fix that)

But he suggested it only if you really cannot stand the loss of efficiency that comes with recalculating that value. Sometimes you need to make such a trade-off.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
June 15 2014 15:46 GMT
#9792
--- Nuked ---
mcc
Profile Joined October 2010
Czech Republic4646 Posts
June 15 2014 15:53 GMT
#9793
On June 16 2014 00:46 Nesserev wrote:
Show nested quote +
On June 16 2014 00:24 mcc wrote:
But he suggested it only if you really cannot stand the loss of efficiency that comes with recalculating that value. Sometimes you need to make such a trade-off.

What trade-off is there to make, when one option is always right, and the other one is easily wrong?
Seriously, two people in a row that don't bother to read what I pointed out?

Always calculating the value actually has costs you know. So if those costs are too big for your particular scenario (not likely in the case of Person class, but possible) you might want to sacrifice the safety for the efficiency.
spinesheath
Profile Blog Joined June 2009
Germany8679 Posts
June 15 2014 16:02 GMT
#9794
Well, I did miss the lazy loading part. My bad. I wouldn't even consider that though. That's the kind of optimization that would pretty much never improve performance, considering how fast the calculation is.
If you have a good reason to disagree with the above, please tell me. Thank you.
mcc
Profile Joined October 2010
Czech Republic4646 Posts
June 15 2014 16:11 GMT
#9795
On June 16 2014 01:02 spinesheath wrote:
Well, I did miss the lazy loading part. My bad. I wouldn't even consider that though. That's the kind of optimization that would pretty much never improve performance, considering how fast the calculation is.

It would, but the exact amount would depend on what exactly are you calculating. On one hand you have boolean test and a retrieval of int/struct. On the other hand you have date subtraction, which is extremely annoying operation if you are not dealing with pure ticks1 - ticks2 scenario.

But I agree that it makes no sense in nearly any practical scenario that i can conceive of.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
June 15 2014 16:48 GMT
#9796
--- Nuked ---
mcc
Profile Joined October 2010
Czech Republic4646 Posts
June 15 2014 17:33 GMT
#9797
On June 16 2014 01:48 Nesserev wrote:
Sacrificing safety for efficiency, is Machiavellian logic. What're you going to do with all those ticks that you saved if you're not even sure if you're working with the information that you want?

If you're concerned about efficiency on a certain level, but 'solving' that efficiency problem could possibly compromise the correctness of the information that you're working with, then it's better to try to solve that efficiency problem on a different level within your code structure: maybe the code on a lower level can be made more efficient, maybe the code on a higher level can minimize your efficiency concern (f.e: calling getAge() once, and storing it in a variable; instead of calling getAge() multiple times within the same runcycle). Otherwise though luck.

Efficiency is only your prime goal, when you're working with lossy information, like livestreams, or when speed is critical, and you can assure that the 'loss of precision' of information has no impact.

No. It is not either or. It is a tradeoff on a continuous scale. The code that uses the lazy evaluation can and most likely will be as correct as the one without. It just won't be as elegant and thus more bug-prone. That is the trade-off. It is not between efficiency and correctness, it between efficiency and robustness/readability/being error-prone. And sometimes you need to take that trade-off. You are not saying anything that was no accounted for in my previous post, since I just claimed that there are situations where you need to make that trade-off.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
June 15 2014 18:38 GMT
#9798
--- Nuked ---
sluggaslamoo
Profile Blog Joined November 2009
Australia4494 Posts
Last Edited: 2014-06-16 05:53:45
June 16 2014 05:34 GMT
#9799
On June 15 2014 23:42 Nesserev wrote:
Show nested quote +
On June 15 2014 19:21 sluggaslamoo wrote:
You would have a function called Age() which would be called everytime you want to access a persons age. The performance impact should be negligible, however if you wanted to optimise you could use lazy loading.

In my opinion, storing age is rather redundant (if you already have the birthdate), and potentially 'dangerous'.

Storing age might actually not be a good idea, since age is based on the difference between the current date and the date of birth, and the current date changes every day. Imagine if the age-variable of a person is initiated the day before his birthday... the next day his age won't be correct anymore. Of course, you can have mechanisms in place to keep track of whether or not the age is still correct, but that would totally negate the advantages of 'lazy loading' over a simple function that calculates the age (which will always be correct).


Wow trust the programming thread to completely take me completely out of context.

On June 15 2014 19:21 sluggaslamoo wrote:
You would have a function called Age() which would be called everytime you want to access a persons age. The performance impact should be negligible, however if you wanted to optimise you could use lazy loading.

As for method size, 99.999% of the time there is a way to split it up into smaller methods. If you know how to do it properly it will always make the code more maintainable and neater.


This is extremely vague upon rereading actually, sorry. It makes more sense if you read the post above it which talked about writing a getter, I removed the "get" and implied that age would be calculated every time, and yes most of the time it is not necessary to store age.

BUT IF (A BIG IF) performance is an issue!!!! for example if you are working on a database warehouse or caching system on a site that uses a ton of sorting and recalculating with millions of users, then you could lazily evaluate it.

Sorry for providing alternatives jeez

Cheers mcc for covering up for me though.

On June 16 2014 03:38 Nesserev wrote:
Show nested quote +
On June 16 2014 02:33 mcc wrote:
No. It is not either or. It is a tradeoff on a continuous scale. The code that uses the lazy evaluation can and most likely will be as correct as the one without. It just won't be as elegant and thus more bug-prone. That is the trade-off. It is not between efficiency and correctness, it between efficiency and robustness/readability/being error-prone. And sometimes you need to take that trade-off. You are not saying anything that was no accounted for in my previous post, since I just claimed that there are situations where you need to make that trade-off.

Ow... I think we were discussing two different things in the end. You were focusing on the fact that you sometimes have to make the trade-off between efficiency and code clarity. I was talking about that one shouldn't take the more 'efficient' way, if it could lead to errors due to the chosen strategy, like the 'lazy loading' of age example.


Only in a utopian world will you be able to write code while not worrying about performance. If you work for a site with tonnes of users, you will have to make "sacrifices" sometimes.

Yes in the beginning you will not pre-optimise, but after load testing if you know for a fact that the new feature you coded will not fly in production you will have to add extra code to optimise it. Yes it makes the code more complex, and harder to maintain, but its better than a site that doesn't run.

Ops can also be stubborn or lazy as hell too, sometimes they can't just boot up more powerful servers even though it would be cheaper overall because it means talking to the higher-ups for a bigger budget. But sometimes the thing you wrote is so slow, that its worth sacrificing a days work to make the whole site run faster.

Yes this probably wouldn't apply to a method like Age(), but definitely database and sorting algorithms sometimes end up being more complicated than necessary purely for performance reasons. Especially with mapped relational database libraries, they are less than ideal when it comes to efficiency.
Come play Android Netrunner - http://www.teamliquid.net/forum/viewmessage.php?topic_id=409008
Ghin
Profile Blog Joined January 2005
United States2391 Posts
June 16 2014 06:08 GMT
#9800
I tried learning Python on whatever sites were on the first page of Google. I spent a couple hours every day for 3 weeks and never really got to a point I could do anything except add simple numbers and look stuff up in documentation. Oh, also I can put lots of semicolons and parenthesis in.

To be honest, this thread isn't at all helpful for someone who knows nothing about programming. It's really hard to look through 500 pages and try to guess which of you are bronze league retarded theorycrafters when I have no knowledge of programming to help me decide that. I don't think it was intended to be helpful to noobs, so I'm fine with that.

It seems like none of the sites are focused towards making some simple games, which is what I want to do with it. If I could get to the point where my drawings are up on the screen, I think that might motivate me more. At this point, I don't know if it would be better to keep slamming my head into the wall trying to figure it out or just give up and find some random nob to work with.
Legalize drugs and murder.
Prev 1 488 489 490 491 492 1032 Next
Please log in or register to reply.
Live Events Refresh
Replay Cast
00:00
WardiTV Mondays #62
CranKy Ducklings144
LiquipediaDiscussion
[ Submit Event ]
Live Streams
Refresh
StarCraft 2
Nathanias 193
StarCraft: Brood War
Artosis 760
Bale 76
Tasteless 36
Dota 2
monkeys_forever880
NeuroSwarm136
League of Legends
JimRising 623
C9.Mang0334
Counter-Strike
Fnx 50
Other Games
summit1g13926
tarik_tv5172
Day[9].tv799
shahzam572
Maynarde161
Mew2King91
ViBE55
CosmosSc2 44
ToD41
Organizations
Other Games
gamesdonequick1018
StarCraft 2
Blizzard YouTube
StarCraft: Brood War
BSLTrovo
sctven
[ Show 14 non-featured ]
StarCraft 2
• Hupsaiya 83
• AfreecaTV YouTube
• intothetv
• Kozan
• IndyKCrew
• LaughNgamezSOOP
• Migwel
• sooper7s
StarCraft: Brood War
• RayReign 28
• BSLYoutube
• STPLYoutube
• ZZZeroYoutube
League of Legends
• Doublelift4950
Other Games
• Day9tv799
Upcoming Events
The PondCast
7h 3m
OSC
13h 3m
Demi vs Mixu
Nicoract vs TBD
Babymarine vs MindelVK
ForJumy vs TBD
Shameless vs Percival
Replay Cast
21h 3m
Korean StarCraft League
2 days
CranKy Ducklings
2 days
WardiTV 2025
2 days
SC Evo League
2 days
BSL 21
2 days
Sziky vs OyAji
Gypsy vs eOnzErG
OSC
2 days
Solar vs Creator
ByuN vs Gerald
Percival vs Babymarine
Moja vs Krystianer
EnDerr vs ForJumy
sebesdes vs Nicoract
Sparkling Tuna Cup
3 days
[ Show More ]
WardiTV 2025
3 days
OSC
3 days
BSL 21
3 days
Bonyth vs StRyKeR
Tarson vs Dandy
Replay Cast
4 days
Wardi Open
4 days
StarCraft2.fi
4 days
Monday Night Weeklies
4 days
Replay Cast
4 days
WardiTV 2025
5 days
StarCraft2.fi
5 days
PiGosaur Monday
5 days
StarCraft2.fi
6 days
Tenacious Turtle Tussle
6 days
Liquipedia Results

Completed

Proleague 2025-11-30
RSL Revival: Season 3
Light HT

Ongoing

C-Race Season 1
IPSL Winter 2025-26
KCM Race Survival 2025 Season 4
YSL S2
BSL Season 21
CSCL: Masked Kings S3
Slon Tour Season 2
Acropolis #4 - TS3
META Madness #9
SL Budapest Major 2025
ESL Impact League Season 8
BLAST Rivals Fall 2025
IEM Chengdu 2025
PGL Masters Bucharest 2025
Thunderpick World Champ.
CS Asia Championships 2025
ESL Pro League S22
StarSeries Fall 2025
FISSURE Playground #2

Upcoming

BSL 21 Non-Korean Championship
Acropolis #4
IPSL Spring 2026
Bellum Gens Elite Stara Zagora 2026
HSC XXVIII
RSL Offline Finals
WardiTV 2025
Kuram Kup
PGL Cluj-Napoca 2026
IEM Kraków 2026
BLAST Bounty Winter 2026
BLAST Bounty Winter Qual
eXTREMESLAND 2025
TLPD

1. ByuN
2. TY
3. Dark
4. Solar
5. Stats
6. Nerchio
7. sOs
8. soO
9. INnoVation
10. Elazer
1. Rain
2. Flash
3. EffOrt
4. Last
5. Bisu
6. Soulkey
7. Mini
8. Sharp
Sidebar Settings...

Advertising | Privacy Policy | Terms Of Use | Contact Us

Original banner artwork: Jim Warren
The contents of this webpage are copyright © 2025 TLnet. All Rights Reserved.