• Log InLog In
  • Register
Liquid`
Team Liquid Liquipedia
EDT 16:23
CEST 22:23
KST 05:23
  • 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
Code S Season 1 - RO8 Preview5[ASL21] Ro8 Preview Pt2: Progenitors8Code S Season 1 - RO12 Group A: Rogue, Percival, Solar, Zoun13[ASL21] Ro8 Preview Pt1: Inheritors16[ASL21] Ro16 Preview Pt2: All Star10
Community News
Maestros of The Game 2 announcement and schedule !7Weekly Cups (April 27-May 4): Clem takes triple0RSL Revival: Season 5 - Qualifiers and Main Event12Code S Season 1 (2026) - RO12 Results12026 GSL Season 1 Qualifiers25
StarCraft 2
General
Code S Season 1 - RO8 Preview Behind the Blue - Team Liquid History Book Weekly Cups (April 27-May 4): Clem takes triple Blizzard Classic Cup @ BlizzCon 2026 - $100k prize pool Code S Season 1 (2026) - RO12 Results
Tourneys
Sparkling Tuna Cup - Weekly Open Tournament Sea Duckling Open (Global, Bronze-Diamond) Maestros of The Game 2 announcement and schedule ! GSL Code S Season 1 (2026) RSL Revival: Season 5 - Qualifiers and Main Event
Strategy
Custom Maps
[D]RTS in all its shapes and glory <3 [A] Nemrods 1/4 players
External Content
Mutation # 524 Death and Taxes The PondCast: SC2 News & Results Mutation # 523 Firewall Mutation # 522 Flip My Base
Brood War
General
Quality of life changes in BW that you will like ? BGH Auto Balance -> http://bghmmr.eu/ RepMastered™: replay sharing and analyzer site Tulbo's ASL S21 Ro8 Post-Review Why there arent any 256x256 pro maps?
Tourneys
[ASL21] Ro8 Day 4 Escore Tournament StarCraft Season 2 [Megathread] Daily Proleagues Small VOD Thread 2.0
Strategy
Simple Questions, Simple Answers Fighting Spirit mining rates What's the deal with APM & what's its true value Any training maps people recommend?
Other Games
General Games
Stormgate/Frost Giant Megathread Nintendo Switch Thread Daigo vs Menard Best of 10 Path of Exile OutLive 25 (RTS Game)
Dota 2
The Story of Wings Gaming
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
Vanilla Mini Mafia Mafia Game Mode Feedback/Ideas TL Mafia Community Thread Five o'clock TL Mafia
Community
General
US Politics Mega-thread The Letting Off Steam Thread European Politico-economics QA Mega-thread UK Politics Mega-thread Canadian Politics Mega-thread
Fan Clubs
The IdrA Fan Club
Media & Entertainment
Anime Discussion Thread [Manga] One Piece [Req][Books] Good Fantasy/SciFi books
Sports
2024 - 2026 Football Thread McBoner: A hockey love story Formula 1 Discussion
World Cup 2022
Tech Support
streaming software Strange computer issues (software) [G] How to Block Livestream Ads
TL Community
The Automated Ban List
Blogs
How EEG Data Can Predict Gam…
TrAiDoS
ramps on octagon
StaticNine
Funny Nicknames
LUCKY_NOOB
Customize Sidebar...

Website Feedback

Closed Threads



Active: 1215 users

The Big Programming Thread - Page 492

Forum Index > General Forum
Post a Reply
Prev 1 490 491 492 493 494 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.
CorsairHero
Profile Joined December 2008
Canada9491 Posts
June 19 2014 07:50 GMT
#9821
Any recommendations for courses/sites to look at for project/product management for a software feature?
© Current year.
Ethenielle
Profile Blog Joined December 2005
Norway1006 Posts
Last Edited: 2014-06-19 18:06:28
June 19 2014 18:05 GMT
#9822
After reading about the observer pattern, I think that's pretty much what I'm doing now.

Anyway, for the class that I expose to the user it looks something like this:


class PointLight
{
int radius;
float pos[3];
[..] etc

vector<Callback> callbacks;

public:
void AddCallback(...);
void X(float x)
{
// set x
// trigger callbacks
}

};


Then whenever the user adds a light to the scene, the engine would create an internal PointLight class which stores additional information that only the engine needs to know about, and it would also register a callback with the exposed class. Something like this:


struct PointLight_Internal
{
PointLight* exposed;
DirectX::XMFLOAT4X4 viewTransform;
ShadowMapOmni smap;
[...]

void OnPropertyChanged(...); // This would recalculate dependent variables such as viewTransform when position changes
};

void Engine::AddLight(PointLight& pl)
{
PointLight_Internal pli;
pli.exposed = &pl;
pl.AddCallback(pli.OnPropertyChanged)

lights.push_back(pli);
}


And then the rendering would obviously only operate on the internal class. So then I get the benefit of hiding these engine-only relevant variables from the user, but in their stead there is this "mysterious" array of callbacks. Ideally, the class I expose to the user would just be a POD class. Maybe that's not possible without constantly polling/updating dependent variables, though. I hope that was maybe a bit clearer explanation of my problem...?
Theres a fine line between fishing and just standing on the shore like an idiot.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
Last Edited: 2014-06-19 22:12:56
June 19 2014 21:56 GMT
#9823
--- Nuked ---
mcc
Profile Joined October 2010
Czech Republic4646 Posts
June 20 2014 08:45 GMT
#9824
On June 20 2014 03:05 Ethenielle wrote:
After reading about the observer pattern, I think that's pretty much what I'm doing now.

Anyway, for the class that I expose to the user it looks something like this:


class PointLight
{
int radius;
float pos[3];
[..] etc

vector<Callback> callbacks;

public:
void AddCallback(...);
void X(float x)
{
// set x
// trigger callbacks
}

};


Then whenever the user adds a light to the scene, the engine would create an internal PointLight class which stores additional information that only the engine needs to know about, and it would also register a callback with the exposed class. Something like this:


struct PointLight_Internal
{
PointLight* exposed;
DirectX::XMFLOAT4X4 viewTransform;
ShadowMapOmni smap;
[...]

void OnPropertyChanged(...); // This would recalculate dependent variables such as viewTransform when position changes
};

void Engine::AddLight(PointLight& pl)
{
PointLight_Internal pli;
pli.exposed = &pl;
pl.AddCallback(pli.OnPropertyChanged)

lights.push_back(pli);
}


And then the rendering would obviously only operate on the internal class. So then I get the benefit of hiding these engine-only relevant variables from the user, but in their stead there is this "mysterious" array of callbacks. Ideally, the class I expose to the user would just be a POD class. Maybe that's not possible without constantly polling/updating dependent variables, though. I hope that was maybe a bit clearer explanation of my problem...?

I am very confused by this design. Why don't you have just one Light class and hide what you want to hide by making it private ? Plus ideally make that class immutable, or at least clone it on Engine:AddLight if you really need to.
Ethenielle
Profile Blog Joined December 2005
Norway1006 Posts
June 20 2014 15:24 GMT
#9825
Initially that's what I did. But the engine needs access to those private variables for rendering, so then I had to friend the engine. I didn't like that, so that's why I ended up splitting the class in two, and that leads to the whole callback messy situation, which isn't good either.

On June 20 2014 06:56 Nesserev wrote:
It might be a bit rude even though I mean no offense, and it's only based on a hunch, but do you even understand how a graphics engine is supposed to work?

EDIT: Now that I think about it, asking this question doesn't help in any way... from what you've described about your engine so far, it just seems that the design of your engine is rather flawed.


Great advice, thanks. You're the reason I ask for help.
Theres a fine line between fishing and just standing on the shore like an idiot.
mcc
Profile Joined October 2010
Czech Republic4646 Posts
Last Edited: 2014-06-20 17:56:00
June 20 2014 17:54 GMT
#9826
On June 21 2014 00:24 Ethenielle wrote:
Initially that's what I did. But the engine needs access to those private variables for rendering, so then I had to friend the engine. I didn't like that, so that's why I ended up splitting the class in two, and that leads to the whole callback messy situation, which isn't good either.

What I meant that maybe you should have the Light structure just as POD structure. The whole concept of reacting to changes that the user makes later seems flawed. User should just create the Light structure, call AddLight by passing the structure in and that is it. Your engine should not react to changes to that struct that happen after the call to AddLight. You can achieve it either by cloning or by making the struct immutable. Another way would be to just not expose any class at all and just expose AddLight and pass all necessary info in parameters.

If you want to allow user to change the Light structure later, you can just return from AddLight a handle that he can later use in SetLight_x(handle, value) or Setx(typedHandle, value). You avoid dealing with all kinds of nasty surprises by more strictly controlling change to those variables.
urboss
Profile Joined September 2013
Austria1223 Posts
June 20 2014 18:06 GMT
#9827
On June 19 2014 01:41 Ethenielle wrote:
That's what I'm doing, but the internal classes have to know whenever the user changes something in the public class, thus the callbacks. Which I don't like.. the user shouldn't know about that, either.

Why not use an abstract class and derive the public class from it?
nunez
Profile Blog Joined February 2011
Norway4003 Posts
Last Edited: 2014-06-20 21:41:34
June 20 2014 21:36 GMT
#9828
@ethenielle
if you want a good case study of an interface using callbacks you can peep boost asio! in f.ex async_read they take the callback functor as a templated argument with a pre-specced signature:

template<
typename AsyncReadStream,
typename Allocator,
typename CompletionCondition,
typename ReadHandler>
void-or-deduced async_read(
AsyncReadStream & s,
basic_streambuf< Allocator > & b,
CompletionCondition completion_condition,
ReadHandler handler
);

void handler(
const boost::system::error_code& error,
std::size_t bytes_transferred
);

that alleviates the need for the 'exposed class' and instead you ask the user to provide a functor satisfying your signature.
conspired against by a confederacy of dunces.
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
June 20 2014 23:29 GMT
#9829
--- Nuked ---
Shield
Profile Blog Joined August 2009
Bulgaria4824 Posts
June 21 2014 01:43 GMT
#9830
Has anyone got any experience with Java 8? Is there anything that is 'must learn'? I've seen lambda expressions, but they seem weird to me; not verbose enough.
phar
Profile Joined August 2011
United States1080 Posts
Last Edited: 2014-06-21 01:52:12
June 21 2014 01:46 GMT
#9831
^

As a meta point in software development, it is good to develop a thick skin to questions that seem overly blunt. A lot of people aren't trying to be mean, they just might not present things in super nice ways. People are also often very willing to teach and explain things, because helping other people understand how things work is the only way any shit gets done in software. Modern software is way too big and complex for any one person to do alone.

Sometimes a question "do you know how X works?" is literally just that - a query to see how much knowledge/experience you have on a specific topic, so people can help you learn more.

(Of course sometimes people can take it too far)

On June 21 2014 10:43 darkness wrote:
Has anyone got any experience with Java 8? Is there anything that is 'must learn'? I've seen lambda expressions, but they seem weird to me; not verbose enough.

No direct experience yet, because my company has yet to bless java 8 for production environments, and I haven't had any time to dick around with experimental stuff.

That said, can't wait for lambda expressions. Gonna remove a lot of bullshit anonymous classes that you have to hack in to do stuff that shouldn't be super hard to write. Also interested to see what tooling I'm gonna have access to around type annotations, though my guess is it's gonna be years out for me


I was just joking with a co-worker today - "What if the powers that be directing java are secretly haskell fanatics, and by java 12 we suddenly get full blown functional goodness?" http://res.cloudinary.com/readnuts-com/image/upload/c_fit,h_350,w_350/jayr9ifjyl9ztmjflk32.jpg

:\
Who after all is today speaking about the destruction of the Armenians?
Ethenielle
Profile Blog Joined December 2005
Norway1006 Posts
June 21 2014 09:36 GMT
#9832
On June 21 2014 03:06 urboss wrote:
Show nested quote +
On June 19 2014 01:41 Ethenielle wrote:
That's what I'm doing, but the internal classes have to know whenever the user changes something in the public class, thus the callbacks. Which I don't like.. the user shouldn't know about that, either.

Why not use an abstract class and derive the public class from it?


I'm not sure how this would help...? The issue is that the engine needs to know about variables the user shouldn't know about.

On June 21 2014 08:29 Nesserev wrote:
Show nested quote +
On June 21 2014 00:24 Ethenielle wrote:
Initially that's what I did. But the engine needs access to those private variables for rendering, so then I had to friend the engine. I didn't like that, so that's why I ended up splitting the class in two, and that leads to the whole callback messy situation, which isn't good either.

On June 20 2014 06:56 Nesserev wrote:
It might be a bit rude even though I mean no offense, and it's only based on a hunch, but do you even understand how a graphics engine is supposed to work?

EDIT: Now that I think about it, asking this question doesn't help in any way... from what you've described about your engine so far, it just seems that the design of your engine is rather flawed.


Great advice, thanks. You're the reason I ask for help.

[...]


It's hard to answer because it's the 3846th time I get an unqualified "your code sucks" without any further help.

I have no previous experience with writing graphics engines, I spent a couple of months reading through Frank D Luna's Game Programming with DirectX 11, which was ok for getting started I suppose but it had so many holes I spent an equal amount of time just googling stuff. Anyway, that's that. If I knew exactly how to craft a graphics engine I probably wouldn't be asking.

Anyway, we went from my code sucks to callbacks and internal classes suck. Could you help me with how I am supposed to design the public interface for say a light or sprite?

mcc

Not sure I am a fan of having setters for every property directly in the engine, I mean from a user point of view it would be hard to find the relevant setters for a given class, and from a maintenance point of view you suddenly have a colossal class chock full of irrelevant setters/getters. And having looked around at various other engines, they all seem to use a Light/Sprite/whatever class which you can work with in an intuitive manner.

Maybe the "best" solution really is to put all relevant variables in their respective class and friend the engine...? Even though friend is anathema.

Anyway, I appreciate you guys taking the time to help me out. Thanks!
Theres a fine line between fishing and just standing on the shore like an idiot.
spinesheath
Profile Blog Joined June 2009
Germany8679 Posts
Last Edited: 2014-06-21 09:46:37
June 21 2014 09:46 GMT
#9833
On June 21 2014 18:36 Ethenielle wrote:
Maybe the "best" solution really is to put all relevant variables in their respective class and friend the engine...? Even though friend is anathema.

Friend is one of the better options C++ has to offer in terms of encapsulation. You shouldn't overuse it and you should know damn well what you are doing, like with everything else C++. But there really is no reason to hate on friend.

Whether it's the right tool for you task, I don't know. Didn't do much work in that area.
If you have a good reason to disagree with the above, please tell me. Thank you.
Manit0u
Profile Blog Joined August 2004
Poland17743 Posts
June 21 2014 10:54 GMT
#9834
On June 21 2014 10:46 phar wrote:
^

As a meta point in software development, it is good to develop a thick skin to questions that seem overly blunt. A lot of people aren't trying to be mean, they just might not present things in super nice ways. People are also often very willing to teach and explain things, because helping other people understand how things work is the only way any shit gets done in software. Modern software is way too big and complex for any one person to do alone.

Sometimes a question "do you know how X works?" is literally just that - a query to see how much knowledge/experience you have on a specific topic, so people can help you learn more.

(Of course sometimes people can take it too far)

Show nested quote +
On June 21 2014 10:43 darkness wrote:
Has anyone got any experience with Java 8? Is there anything that is 'must learn'? I've seen lambda expressions, but they seem weird to me; not verbose enough.

No direct experience yet, because my company has yet to bless java 8 for production environments, and I haven't had any time to dick around with experimental stuff.

That said, can't wait for lambda expressions. Gonna remove a lot of bullshit anonymous classes that you have to hack in to do stuff that shouldn't be super hard to write. Also interested to see what tooling I'm gonna have access to around type annotations, though my guess is it's gonna be years out for me


I was just joking with a co-worker today - "What if the powers that be directing java are secretly haskell fanatics, and by java 12 we suddenly get full blown functional goodness?" http://res.cloudinary.com/readnuts-com/image/upload/c_fit,h_350,w_350/jayr9ifjyl9ztmjflk32.jpg

:\


Screw Haskell, LISP is where it's at
Time is precious. Waste it wisely.
Shield
Profile Blog Joined August 2009
Bulgaria4824 Posts
Last Edited: 2014-06-21 14:28:39
June 21 2014 14:23 GMT
#9835
What do you think of the idea to have a separate controller (listener) for each GUI component? Is it a good idea? On top of that, I've implemented a controller factory which just accepts a class, then responds by returning the appropriate listener.

So for example:


public ActionListener getRelevantController(Class<?> theClass) {
if (theClass == SomeClass.class)
return new SomeClassListener(arg1);
else if (theClass == SomeOtherClass.class)
return new SomeOtherClassListener(arg1, arg2);
else
throw new IllegalArgumentException(<error here>);
}


I'm just wondering if I'm not doing some major bullshit. It's my own project, so I have no constraints about what I code.

Edit: I know the classic Factory pattern usually involves a static method but I decided not to pass arg1 and arg2 every time, and not every listener needs both at the same time.
bangsholt
Profile Joined June 2011
Denmark138 Posts
June 22 2014 10:52 GMT
#9836
wat

Have you heard of MVC?

One view has one controller.

One controller can have more views.

Why would you have a controller per component? If it's because you have overlap in usage, you can always extend on a base class that provides the shared functionality.
billy5000
Profile Blog Joined December 2010
United States865 Posts
June 24 2014 01:56 GMT
#9837
Got a question about eclipse. We have a workspace that consists of many projects, and there's a bit of a delay. Ideally, I would like to load a single project on eclipse rather than loading all of them. So instead of the root being the workspace, it would be the project. Is there a way to do this?
Tiger got to hunt, bird got to fly; Man got to sit and wonder, 'Why, why, why?' Tiger got to sleep, bird got to land; Man got to tell himself he understand. Vonnegut
Nesserev
Profile Blog Joined January 2011
Belgium2760 Posts
Last Edited: 2014-06-24 03:14:25
June 24 2014 03:11 GMT
#9838
--- Nuked ---
berated-
Profile Blog Joined February 2007
United States1134 Posts
June 24 2014 11:32 GMT
#9839
On June 24 2014 10:56 billy5000 wrote:
Got a question about eclipse. We have a workspace that consists of many projects, and there's a bit of a delay. Ideally, I would like to load a single project on eclipse rather than loading all of them. So instead of the root being the workspace, it would be the project. Is there a way to do this?


Where is the "bit of delay" you are trying to get rid of?

Sorry if you have already tried this, or if this isn't where your delay is, but, have you tried closing the projects you aren't using? I've had 75+ projects in a workspace and if a lot of the projects were open it would bog it down, but if you just close the projects you aren't using then everything was fast.
billy5000
Profile Blog Joined December 2010
United States865 Posts
June 24 2014 22:34 GMT
#9840
On June 24 2014 20:32 berated- wrote:
Show nested quote +
On June 24 2014 10:56 billy5000 wrote:
Got a question about eclipse. We have a workspace that consists of many projects, and there's a bit of a delay. Ideally, I would like to load a single project on eclipse rather than loading all of them. So instead of the root being the workspace, it would be the project. Is there a way to do this?


Where is the "bit of delay" you are trying to get rid of?

Sorry if you have already tried this, or if this isn't where your delay is, but, have you tried closing the projects you aren't using? I've had 75+ projects in a workspace and if a lot of the projects were open it would bog it down, but if you just close the projects you aren't using then everything was fast.


I didn't know about closing projects, thanks!
Tiger got to hunt, bird got to fly; Man got to sit and wonder, 'Why, why, why?' Tiger got to sleep, bird got to land; Man got to tell himself he understand. Vonnegut
Prev 1 490 491 492 493 494 1032 Next
Please log in or register to reply.
Live Events Refresh
BSL
19:00
RO16 Group C
Artosis vs TerrOr
spx vs StRyKeR
ZZZero.O370
LiquipediaDiscussion
[ Submit Event ]
Live Streams
Refresh
StarCraft 2
mouzHeroMarine 465
BRAT_OK 56
CosmosSc2 50
Vindicta 26
PattyMac 13
StarCraft: Brood War
Calm 3918
Mini 404
ZZZero.O 370
ggaemo 232
firebathero 152
Dewaltoss 135
Backho 37
Hyun 28
NaDa 4
Dota 2
monkeys_forever1056
LuMiX1
League of Legends
Doublelift1749
JimRising 133
Reynor67
Counter-Strike
Fnx 1896
Heroes of the Storm
Liquid`Hasu372
Khaldor228
Other Games
Grubby4970
FrodaN2549
fl0m1250
KnowMe265
RotterdaM228
Hui .189
kaitlyn45
Organizations
Other Games
gamesdonequick3519
StarCraft 2
angryscii 18
Blizzard YouTube
StarCraft: Brood War
BSLTrovo
[ Show 23 non-featured ]
StarCraft 2
• Hupsaiya 69
• printf 59
• davetesta28
• Adnapsc2 22
• Response 2
• IndyKCrew
• sooper7s
• AfreecaTV YouTube
• Migwel
• intothetv
• LaughNgamezSOOP
• Kozan
StarCraft: Brood War
• FirePhoenix10
• STPLYoutube
• ZZZeroYoutube
• BSLYoutube
Dota 2
• WagamamaTV913
• masondota2739
• lizZardDota280
League of Legends
• imaqtpie2115
• TFBlade918
Other Games
• Shiphtur269
• tFFMrPink 20
Upcoming Events
Replay Cast
3h 37m
Sparkling Tuna Cup
13h 37m
RSL Revival
13h 37m
Cure vs Zoun
Clem vs Lambo
WardiTV Invitational
15h 37m
ByuN vs Rogue
Solar vs Ryung
Zoun vs Percival
Cure vs SHIN
BSL
22h 37m
Dewalt vs DragOn
Aether vs Jimin
GSL
1d 11h
Afreeca Starleague
1d 13h
Soma vs Leta
Wardi Open
1d 15h
Monday Night Weeklies
1d 19h
OSC
2 days
[ Show More ]
CranKy Ducklings
2 days
Afreeca Starleague
2 days
Light vs Flash
Replay Cast
3 days
Replay Cast
4 days
The PondCast
4 days
Replay Cast
5 days
RSL Revival
5 days
Korean StarCraft League
6 days
RSL Revival
6 days
BSL
6 days
Liquipedia Results

Completed

Escore Tournament S2: W6
WardiTV TLMC #16
Nations Cup 2026

Ongoing

BSL Season 22
ASL Season 21
CSL 2026 SPRING (S20)
IPSL Spring 2026
KCM Race Survival 2026 Season 2
Acropolis #4
KK 2v2 League Season 1
SCTL 2026 Spring
RSL Revival: Season 5
2026 GSL S1
PGL Astana 2026
BLAST Rivals Spring 2026
IEM Rio 2026
PGL Bucharest 2026
Stake Ranked Episode 1
BLAST Open Spring 2026
ESL Pro League S23 Finals
ESL Pro League S23 Stage 1&2

Upcoming

BSL 22 Non-Korean Championship
YSL S3
Escore Tournament S2: W7
Escore Tournament S2: W8
CSLAN 4
Kung Fu Cup 2026 Grand Finals
HSC XXIX
uThermal 2v2 2026 Main Event
Maestros of the Game 2
2026 GSL S2
BLAST Bounty Summer 2026: Closed Qualifier
Stake Ranked Episode 3
XSE Pro League 2026
IEM Cologne Major 2026
Stake Ranked Episode 2
CS Asia Championships 2026
Asian Champions League 2026
IEM Atlanta 2026
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 © 2026 TLnet. All Rights Reserved.